2007-07-11 21:01:13 +04:00
|
|
|
//===--- ASTContext.cpp - Context to hold long-lived AST nodes ------------===//
|
|
|
|
//
|
|
|
|
// The LLVM Compiler Infrastructure
|
|
|
|
//
|
2007-12-29 22:59:25 +03:00
|
|
|
// This file is distributed under the University of Illinois Open Source
|
|
|
|
// License. See LICENSE.TXT for details.
|
2007-07-11 21:01:13 +04:00
|
|
|
//
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
//
|
|
|
|
// This file implements the ASTContext interface.
|
|
|
|
//
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
|
|
|
|
#include "clang/AST/ASTContext.h"
|
2009-12-22 17:23:30 +03:00
|
|
|
#include "clang/AST/CharUnits.h"
|
2008-08-08 00:55:28 +04:00
|
|
|
#include "clang/AST/DeclCXX.h"
|
2007-10-01 23:00:59 +04:00
|
|
|
#include "clang/AST/DeclObjC.h"
|
2009-02-04 22:02:06 +03:00
|
|
|
#include "clang/AST/DeclTemplate.h"
|
2009-08-19 05:27:32 +04:00
|
|
|
#include "clang/AST/TypeLoc.h"
|
2008-08-11 08:54:23 +04:00
|
|
|
#include "clang/AST/Expr.h"
|
2010-05-20 05:18:31 +04:00
|
|
|
#include "clang/AST/ExprCXX.h"
|
2009-04-10 02:27:44 +04:00
|
|
|
#include "clang/AST/ExternalASTSource.h"
|
2010-10-24 21:26:36 +04:00
|
|
|
#include "clang/AST/ASTMutationListener.h"
|
2009-07-18 23:43:29 +04:00
|
|
|
#include "clang/AST/RecordLayout.h"
|
2011-01-13 21:57:25 +03:00
|
|
|
#include "clang/AST/Mangle.h"
|
2009-06-14 05:54:56 +04:00
|
|
|
#include "clang/Basic/Builtins.h"
|
2009-03-28 06:45:20 +03:00
|
|
|
#include "clang/Basic/SourceManager.h"
|
2007-07-11 21:01:13 +04:00
|
|
|
#include "clang/Basic/TargetInfo.h"
|
2009-10-24 13:57:09 +04:00
|
|
|
#include "llvm/ADT/SmallString.h"
|
2007-10-29 08:01:08 +03:00
|
|
|
#include "llvm/ADT/StringExtras.h"
|
2009-01-18 09:42:49 +03:00
|
|
|
#include "llvm/Support/MathExtras.h"
|
2009-10-24 13:57:09 +04:00
|
|
|
#include "llvm/Support/raw_ostream.h"
|
2011-07-27 22:41:12 +04:00
|
|
|
#include "llvm/Support/Capacity.h"
|
2010-08-16 07:33:14 +04:00
|
|
|
#include "CXXABI.h"
|
2011-05-17 04:46:38 +04:00
|
|
|
#include <map>
|
2009-07-19 01:19:52 +04:00
|
|
|
|
2007-07-11 21:01:13 +04:00
|
|
|
using namespace clang;
|
|
|
|
|
2010-07-03 04:47:00 +04:00
|
|
|
unsigned ASTContext::NumImplicitDefaultConstructors;
|
|
|
|
unsigned ASTContext::NumImplicitDefaultConstructorsDeclared;
|
2010-07-03 03:41:54 +04:00
|
|
|
unsigned ASTContext::NumImplicitCopyConstructors;
|
|
|
|
unsigned ASTContext::NumImplicitCopyConstructorsDeclared;
|
2011-05-26 00:50:04 +04:00
|
|
|
unsigned ASTContext::NumImplicitMoveConstructors;
|
|
|
|
unsigned ASTContext::NumImplicitMoveConstructorsDeclared;
|
2010-07-03 01:50:04 +04:00
|
|
|
unsigned ASTContext::NumImplicitCopyAssignmentOperators;
|
|
|
|
unsigned ASTContext::NumImplicitCopyAssignmentOperatorsDeclared;
|
2011-05-26 00:50:04 +04:00
|
|
|
unsigned ASTContext::NumImplicitMoveAssignmentOperators;
|
|
|
|
unsigned ASTContext::NumImplicitMoveAssignmentOperatorsDeclared;
|
2010-07-03 00:37:36 +04:00
|
|
|
unsigned ASTContext::NumImplicitDestructors;
|
|
|
|
unsigned ASTContext::NumImplicitDestructorsDeclared;
|
|
|
|
|
2007-07-11 21:01:13 +04:00
|
|
|
enum FloatingRank {
|
2011-10-15 03:23:15 +04:00
|
|
|
HalfRank, FloatRank, DoubleRank, LongDoubleRank
|
2007-07-11 21:01:13 +04:00
|
|
|
};
|
|
|
|
|
2010-06-17 01:09:37 +04:00
|
|
|
void
|
|
|
|
ASTContext::CanonicalTemplateTemplateParm::Profile(llvm::FoldingSetNodeID &ID,
|
|
|
|
TemplateTemplateParmDecl *Parm) {
|
|
|
|
ID.AddInteger(Parm->getDepth());
|
|
|
|
ID.AddInteger(Parm->getPosition());
|
2011-01-05 18:48:55 +03:00
|
|
|
ID.AddBoolean(Parm->isParameterPack());
|
2010-06-17 01:09:37 +04:00
|
|
|
|
|
|
|
TemplateParameterList *Params = Parm->getTemplateParameters();
|
|
|
|
ID.AddInteger(Params->size());
|
|
|
|
for (TemplateParameterList::const_iterator P = Params->begin(),
|
|
|
|
PEnd = Params->end();
|
|
|
|
P != PEnd; ++P) {
|
|
|
|
if (TemplateTypeParmDecl *TTP = dyn_cast<TemplateTypeParmDecl>(*P)) {
|
|
|
|
ID.AddInteger(0);
|
|
|
|
ID.AddBoolean(TTP->isParameterPack());
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (NonTypeTemplateParmDecl *NTTP = dyn_cast<NonTypeTemplateParmDecl>(*P)) {
|
|
|
|
ID.AddInteger(1);
|
2011-01-05 18:48:55 +03:00
|
|
|
ID.AddBoolean(NTTP->isParameterPack());
|
2012-03-07 05:09:33 +04:00
|
|
|
ID.AddPointer(NTTP->getType().getCanonicalType().getAsOpaquePtr());
|
2011-01-19 23:10:05 +03:00
|
|
|
if (NTTP->isExpandedParameterPack()) {
|
|
|
|
ID.AddBoolean(true);
|
|
|
|
ID.AddInteger(NTTP->getNumExpansionTypes());
|
2012-03-07 05:09:33 +04:00
|
|
|
for (unsigned I = 0, N = NTTP->getNumExpansionTypes(); I != N; ++I) {
|
|
|
|
QualType T = NTTP->getExpansionType(I);
|
|
|
|
ID.AddPointer(T.getCanonicalType().getAsOpaquePtr());
|
|
|
|
}
|
2011-01-19 23:10:05 +03:00
|
|
|
} else
|
|
|
|
ID.AddBoolean(false);
|
2010-06-17 01:09:37 +04:00
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
TemplateTemplateParmDecl *TTP = cast<TemplateTemplateParmDecl>(*P);
|
|
|
|
ID.AddInteger(2);
|
|
|
|
Profile(ID, TTP);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
TemplateTemplateParmDecl *
|
|
|
|
ASTContext::getCanonicalTemplateTemplateParmDecl(
|
2011-01-12 12:06:06 +03:00
|
|
|
TemplateTemplateParmDecl *TTP) const {
|
2010-06-17 01:09:37 +04:00
|
|
|
// Check if we already have a canonical template template parameter.
|
|
|
|
llvm::FoldingSetNodeID ID;
|
|
|
|
CanonicalTemplateTemplateParm::Profile(ID, TTP);
|
|
|
|
void *InsertPos = 0;
|
|
|
|
CanonicalTemplateTemplateParm *Canonical
|
|
|
|
= CanonTemplateTemplateParms.FindNodeOrInsertPos(ID, InsertPos);
|
|
|
|
if (Canonical)
|
|
|
|
return Canonical->getParam();
|
|
|
|
|
|
|
|
// Build a canonical template parameter list.
|
|
|
|
TemplateParameterList *Params = TTP->getTemplateParameters();
|
2011-07-23 14:55:15 +04:00
|
|
|
SmallVector<NamedDecl *, 4> CanonParams;
|
2010-06-17 01:09:37 +04:00
|
|
|
CanonParams.reserve(Params->size());
|
|
|
|
for (TemplateParameterList::const_iterator P = Params->begin(),
|
|
|
|
PEnd = Params->end();
|
|
|
|
P != PEnd; ++P) {
|
|
|
|
if (TemplateTypeParmDecl *TTP = dyn_cast<TemplateTypeParmDecl>(*P))
|
|
|
|
CanonParams.push_back(
|
|
|
|
TemplateTypeParmDecl::Create(*this, getTranslationUnitDecl(),
|
2011-03-06 18:48:19 +03:00
|
|
|
SourceLocation(),
|
|
|
|
SourceLocation(),
|
|
|
|
TTP->getDepth(),
|
2010-06-17 01:09:37 +04:00
|
|
|
TTP->getIndex(), 0, false,
|
|
|
|
TTP->isParameterPack()));
|
|
|
|
else if (NonTypeTemplateParmDecl *NTTP
|
2011-01-19 23:10:05 +03:00
|
|
|
= dyn_cast<NonTypeTemplateParmDecl>(*P)) {
|
|
|
|
QualType T = getCanonicalType(NTTP->getType());
|
|
|
|
TypeSourceInfo *TInfo = getTrivialTypeSourceInfo(T);
|
|
|
|
NonTypeTemplateParmDecl *Param;
|
|
|
|
if (NTTP->isExpandedParameterPack()) {
|
2011-07-23 14:55:15 +04:00
|
|
|
SmallVector<QualType, 2> ExpandedTypes;
|
|
|
|
SmallVector<TypeSourceInfo *, 2> ExpandedTInfos;
|
2011-01-19 23:10:05 +03:00
|
|
|
for (unsigned I = 0, N = NTTP->getNumExpansionTypes(); I != N; ++I) {
|
|
|
|
ExpandedTypes.push_back(getCanonicalType(NTTP->getExpansionType(I)));
|
|
|
|
ExpandedTInfos.push_back(
|
|
|
|
getTrivialTypeSourceInfo(ExpandedTypes.back()));
|
|
|
|
}
|
|
|
|
|
|
|
|
Param = NonTypeTemplateParmDecl::Create(*this, getTranslationUnitDecl(),
|
2011-03-08 11:55:46 +03:00
|
|
|
SourceLocation(),
|
|
|
|
SourceLocation(),
|
2011-01-19 23:10:05 +03:00
|
|
|
NTTP->getDepth(),
|
|
|
|
NTTP->getPosition(), 0,
|
|
|
|
T,
|
|
|
|
TInfo,
|
|
|
|
ExpandedTypes.data(),
|
|
|
|
ExpandedTypes.size(),
|
|
|
|
ExpandedTInfos.data());
|
|
|
|
} else {
|
|
|
|
Param = NonTypeTemplateParmDecl::Create(*this, getTranslationUnitDecl(),
|
2011-03-08 11:55:46 +03:00
|
|
|
SourceLocation(),
|
|
|
|
SourceLocation(),
|
2011-01-19 23:10:05 +03:00
|
|
|
NTTP->getDepth(),
|
|
|
|
NTTP->getPosition(), 0,
|
|
|
|
T,
|
|
|
|
NTTP->isParameterPack(),
|
|
|
|
TInfo);
|
|
|
|
}
|
|
|
|
CanonParams.push_back(Param);
|
|
|
|
|
|
|
|
} else
|
2010-06-17 01:09:37 +04:00
|
|
|
CanonParams.push_back(getCanonicalTemplateTemplateParmDecl(
|
|
|
|
cast<TemplateTemplateParmDecl>(*P)));
|
|
|
|
}
|
|
|
|
|
|
|
|
TemplateTemplateParmDecl *CanonTTP
|
|
|
|
= TemplateTemplateParmDecl::Create(*this, getTranslationUnitDecl(),
|
|
|
|
SourceLocation(), TTP->getDepth(),
|
2011-01-05 18:48:55 +03:00
|
|
|
TTP->getPosition(),
|
|
|
|
TTP->isParameterPack(),
|
|
|
|
0,
|
2010-06-17 01:09:37 +04:00
|
|
|
TemplateParameterList::Create(*this, SourceLocation(),
|
|
|
|
SourceLocation(),
|
|
|
|
CanonParams.data(),
|
|
|
|
CanonParams.size(),
|
|
|
|
SourceLocation()));
|
|
|
|
|
|
|
|
// Get the new insert position for the node we care about.
|
|
|
|
Canonical = CanonTemplateTemplateParms.FindNodeOrInsertPos(ID, InsertPos);
|
|
|
|
assert(Canonical == 0 && "Shouldn't be in the map!");
|
|
|
|
(void)Canonical;
|
|
|
|
|
|
|
|
// Create the canonical template template parameter entry.
|
|
|
|
Canonical = new (*this) CanonicalTemplateTemplateParm(CanonTTP);
|
|
|
|
CanonTemplateTemplateParms.InsertNode(Canonical, InsertPos);
|
|
|
|
return CanonTTP;
|
|
|
|
}
|
|
|
|
|
2010-08-16 07:33:14 +04:00
|
|
|
CXXABI *ASTContext::createCXXABI(const TargetInfo &T) {
|
2010-08-22 02:46:04 +04:00
|
|
|
if (!LangOpts.CPlusPlus) return 0;
|
|
|
|
|
2010-08-19 06:18:14 +04:00
|
|
|
switch (T.getCXXABI()) {
|
2010-08-22 02:46:04 +04:00
|
|
|
case CXXABI_ARM:
|
|
|
|
return CreateARMCXXABI(*this);
|
|
|
|
case CXXABI_Itanium:
|
2010-08-16 07:33:14 +04:00
|
|
|
return CreateItaniumCXXABI(*this);
|
2010-08-19 06:18:14 +04:00
|
|
|
case CXXABI_Microsoft:
|
|
|
|
return CreateMicrosoftCXXABI(*this);
|
|
|
|
}
|
2012-01-17 10:56:22 +04:00
|
|
|
llvm_unreachable("Invalid CXXABI type!");
|
2010-08-16 07:33:14 +04:00
|
|
|
}
|
|
|
|
|
2011-09-02 04:18:52 +04:00
|
|
|
static const LangAS::Map *getAddressSpaceMap(const TargetInfo &T,
|
2011-03-19 01:38:29 +03:00
|
|
|
const LangOptions &LOpts) {
|
|
|
|
if (LOpts.FakeAddressSpaceMap) {
|
|
|
|
// The fake address space map must have a distinct entry for each
|
|
|
|
// language-specific address space.
|
|
|
|
static const unsigned FakeAddrSpaceMap[] = {
|
|
|
|
1, // opencl_global
|
|
|
|
2, // opencl_local
|
|
|
|
3 // opencl_constant
|
|
|
|
};
|
2011-09-02 04:18:52 +04:00
|
|
|
return &FakeAddrSpaceMap;
|
2011-03-19 01:38:29 +03:00
|
|
|
} else {
|
2011-09-02 04:18:52 +04:00
|
|
|
return &T.getAddressSpaceMap();
|
2011-03-19 01:38:29 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-09-02 00:23:19 +04:00
|
|
|
ASTContext::ASTContext(LangOptions& LOpts, SourceManager &SM,
|
2011-09-02 04:18:52 +04:00
|
|
|
const TargetInfo *t,
|
2008-08-11 08:54:23 +04:00
|
|
|
IdentifierTable &idents, SelectorTable &sels,
|
2009-06-14 05:54:56 +04:00
|
|
|
Builtin::Context &builtins,
|
2011-09-02 04:18:52 +04:00
|
|
|
unsigned size_reserve,
|
|
|
|
bool DelayInitialization)
|
|
|
|
: FunctionProtoTypes(this_()),
|
|
|
|
TemplateSpecializationTypes(this_()),
|
|
|
|
DependentTemplateSpecializationTypes(this_()),
|
|
|
|
SubstTemplateTemplateParmPacks(this_()),
|
|
|
|
GlobalNestedNameSpecifier(0),
|
|
|
|
Int128Decl(0), UInt128Decl(0),
|
2012-01-17 22:09:05 +04:00
|
|
|
ObjCIdDecl(0), ObjCSelDecl(0), ObjCClassDecl(0), ObjCProtocolClassDecl(0),
|
2011-09-08 05:46:34 +04:00
|
|
|
CFConstantStringTypeDecl(0), ObjCInstanceTypeDecl(0),
|
2011-09-02 04:18:52 +04:00
|
|
|
FILEDecl(0),
|
2011-11-14 01:51:09 +04:00
|
|
|
jmp_bufDecl(0), sigjmp_bufDecl(0), ucontext_tDecl(0),
|
|
|
|
BlockDescriptorType(0), BlockDescriptorExtendedType(0),
|
|
|
|
cudaConfigureCallDecl(0),
|
2011-12-03 04:30:27 +04:00
|
|
|
NullTypeSourceInfo(QualType()),
|
|
|
|
FirstLocalImport(), LastLocalImport(),
|
2011-09-02 04:18:52 +04:00
|
|
|
SourceMgr(SM), LangOpts(LOpts),
|
2011-09-28 02:38:19 +04:00
|
|
|
AddrSpaceMap(0), Target(t), PrintingPolicy(LOpts),
|
2011-09-02 04:18:52 +04:00
|
|
|
Idents(idents), Selectors(sels),
|
|
|
|
BuiltinInfo(builtins),
|
|
|
|
DeclarationNames(*this),
|
2011-09-28 02:38:19 +04:00
|
|
|
ExternalSource(0), Listener(0),
|
2011-09-02 04:18:52 +04:00
|
|
|
LastSDM(0, 0),
|
|
|
|
UniqueBlockByRefTypeID(0)
|
|
|
|
{
|
2009-09-09 19:08:12 +04:00
|
|
|
if (size_reserve > 0) Types.reserve(size_reserve);
|
2008-08-11 08:54:23 +04:00
|
|
|
TUDecl = TranslationUnitDecl::Create(*this);
|
2011-09-02 04:18:52 +04:00
|
|
|
|
|
|
|
if (!DelayInitialization) {
|
|
|
|
assert(t && "No target supplied for ASTContext initialization");
|
|
|
|
InitBuiltinTypes(*t);
|
|
|
|
}
|
2008-08-11 08:54:23 +04:00
|
|
|
}
|
|
|
|
|
2007-07-11 21:01:13 +04:00
|
|
|
ASTContext::~ASTContext() {
|
2010-02-11 10:12:28 +03:00
|
|
|
// Release the DenseMaps associated with DeclContext objects.
|
|
|
|
// FIXME: Is this the ideal solution?
|
|
|
|
ReleaseDeclContextMaps();
|
2010-03-03 02:58:15 +03:00
|
|
|
|
2010-07-25 21:53:33 +04:00
|
|
|
// Call all of the deallocation functions.
|
|
|
|
for (unsigned I = 0, N = Deallocations.size(); I != N; ++I)
|
|
|
|
Deallocations[I].first(Deallocations[I].second);
|
2010-05-23 22:26:36 +04:00
|
|
|
|
2010-06-09 03:00:58 +04:00
|
|
|
// ASTRecordLayout objects in ASTRecordLayouts must always be destroyed
|
2010-07-25 21:53:33 +04:00
|
|
|
// because they can contain DenseMaps.
|
|
|
|
for (llvm::DenseMap<const ObjCContainerDecl*,
|
|
|
|
const ASTRecordLayout*>::iterator
|
|
|
|
I = ObjCLayouts.begin(), E = ObjCLayouts.end(); I != E; )
|
|
|
|
// Increment in loop to prevent using deallocated memory.
|
|
|
|
if (ASTRecordLayout *R = const_cast<ASTRecordLayout*>((I++)->second))
|
|
|
|
R->Destroy(*this);
|
|
|
|
|
2010-06-09 03:00:58 +04:00
|
|
|
for (llvm::DenseMap<const RecordDecl*, const ASTRecordLayout*>::iterator
|
|
|
|
I = ASTRecordLayouts.begin(), E = ASTRecordLayouts.end(); I != E; ) {
|
|
|
|
// Increment in loop to prevent using deallocated memory.
|
|
|
|
if (ASTRecordLayout *R = const_cast<ASTRecordLayout*>((I++)->second))
|
|
|
|
R->Destroy(*this);
|
|
|
|
}
|
2010-08-30 20:49:28 +04:00
|
|
|
|
|
|
|
for (llvm::DenseMap<const Decl*, AttrVec*>::iterator A = DeclAttrs.begin(),
|
|
|
|
AEnd = DeclAttrs.end();
|
|
|
|
A != AEnd; ++A)
|
|
|
|
A->second->~AttrVec();
|
|
|
|
}
|
2009-03-27 02:50:42 +03:00
|
|
|
|
2010-05-23 22:26:36 +04:00
|
|
|
void ASTContext::AddDeallocation(void (*Callback)(void*), void *Data) {
|
|
|
|
Deallocations.push_back(std::make_pair(Callback, Data));
|
|
|
|
}
|
|
|
|
|
2009-09-09 19:08:12 +04:00
|
|
|
void
|
2012-02-05 06:12:40 +04:00
|
|
|
ASTContext::setExternalSource(OwningPtr<ExternalASTSource> &Source) {
|
2009-04-10 02:27:44 +04:00
|
|
|
ExternalSource.reset(Source.take());
|
|
|
|
}
|
|
|
|
|
2007-07-11 21:01:13 +04:00
|
|
|
void ASTContext::PrintStats() const {
|
2011-07-04 09:32:14 +04:00
|
|
|
llvm::errs() << "\n*** AST Context Stats:\n";
|
|
|
|
llvm::errs() << " " << Types.size() << " types total.\n";
|
2009-03-17 02:22:08 +03:00
|
|
|
|
2009-05-26 18:40:08 +04:00
|
|
|
unsigned counts[] = {
|
2009-09-09 19:08:12 +04:00
|
|
|
#define TYPE(Name, Parent) 0,
|
2009-05-26 18:40:08 +04:00
|
|
|
#define ABSTRACT_TYPE(Name, Parent)
|
|
|
|
#include "clang/AST/TypeNodes.def"
|
|
|
|
0 // Extra
|
|
|
|
};
|
2009-04-07 21:20:56 +04:00
|
|
|
|
2007-07-11 21:01:13 +04:00
|
|
|
for (unsigned i = 0, e = Types.size(); i != e; ++i) {
|
|
|
|
Type *T = Types[i];
|
2009-05-26 18:40:08 +04:00
|
|
|
counts[(unsigned)T->getTypeClass()]++;
|
2007-07-11 21:01:13 +04:00
|
|
|
}
|
|
|
|
|
2009-05-26 18:40:08 +04:00
|
|
|
unsigned Idx = 0;
|
|
|
|
unsigned TotalBytes = 0;
|
|
|
|
#define TYPE(Name, Parent) \
|
|
|
|
if (counts[Idx]) \
|
2011-07-04 09:32:14 +04:00
|
|
|
llvm::errs() << " " << counts[Idx] << " " << #Name \
|
|
|
|
<< " types\n"; \
|
2009-05-26 18:40:08 +04:00
|
|
|
TotalBytes += counts[Idx] * sizeof(Name##Type); \
|
|
|
|
++Idx;
|
|
|
|
#define ABSTRACT_TYPE(Name, Parent)
|
|
|
|
#include "clang/AST/TypeNodes.def"
|
2009-09-09 19:08:12 +04:00
|
|
|
|
2011-07-04 09:32:14 +04:00
|
|
|
llvm::errs() << "Total bytes = " << TotalBytes << "\n";
|
|
|
|
|
2010-07-03 00:37:36 +04:00
|
|
|
// Implicit special member functions.
|
2011-07-04 09:32:14 +04:00
|
|
|
llvm::errs() << NumImplicitDefaultConstructorsDeclared << "/"
|
|
|
|
<< NumImplicitDefaultConstructors
|
|
|
|
<< " implicit default constructors created\n";
|
|
|
|
llvm::errs() << NumImplicitCopyConstructorsDeclared << "/"
|
|
|
|
<< NumImplicitCopyConstructors
|
|
|
|
<< " implicit copy constructors created\n";
|
2012-03-11 11:00:24 +04:00
|
|
|
if (getLangOpts().CPlusPlus)
|
2011-07-04 09:32:14 +04:00
|
|
|
llvm::errs() << NumImplicitMoveConstructorsDeclared << "/"
|
|
|
|
<< NumImplicitMoveConstructors
|
|
|
|
<< " implicit move constructors created\n";
|
|
|
|
llvm::errs() << NumImplicitCopyAssignmentOperatorsDeclared << "/"
|
|
|
|
<< NumImplicitCopyAssignmentOperators
|
|
|
|
<< " implicit copy assignment operators created\n";
|
2012-03-11 11:00:24 +04:00
|
|
|
if (getLangOpts().CPlusPlus)
|
2011-07-04 09:32:14 +04:00
|
|
|
llvm::errs() << NumImplicitMoveAssignmentOperatorsDeclared << "/"
|
|
|
|
<< NumImplicitMoveAssignmentOperators
|
|
|
|
<< " implicit move assignment operators created\n";
|
|
|
|
llvm::errs() << NumImplicitDestructorsDeclared << "/"
|
|
|
|
<< NumImplicitDestructors
|
|
|
|
<< " implicit destructors created\n";
|
|
|
|
|
2009-04-10 02:27:44 +04:00
|
|
|
if (ExternalSource.get()) {
|
2011-07-04 09:32:14 +04:00
|
|
|
llvm::errs() << "\n";
|
2009-04-10 02:27:44 +04:00
|
|
|
ExternalSource->PrintStats();
|
|
|
|
}
|
2011-07-04 09:32:14 +04:00
|
|
|
|
2010-07-25 21:53:33 +04:00
|
|
|
BumpAlloc.PrintStats();
|
2007-07-11 21:01:13 +04:00
|
|
|
}
|
|
|
|
|
2011-08-12 10:49:56 +04:00
|
|
|
TypedefDecl *ASTContext::getInt128Decl() const {
|
|
|
|
if (!Int128Decl) {
|
|
|
|
TypeSourceInfo *TInfo = getTrivialTypeSourceInfo(Int128Ty);
|
|
|
|
Int128Decl = TypedefDecl::Create(const_cast<ASTContext &>(*this),
|
|
|
|
getTranslationUnitDecl(),
|
|
|
|
SourceLocation(),
|
|
|
|
SourceLocation(),
|
|
|
|
&Idents.get("__int128_t"),
|
|
|
|
TInfo);
|
|
|
|
}
|
|
|
|
|
|
|
|
return Int128Decl;
|
|
|
|
}
|
|
|
|
|
|
|
|
TypedefDecl *ASTContext::getUInt128Decl() const {
|
|
|
|
if (!UInt128Decl) {
|
|
|
|
TypeSourceInfo *TInfo = getTrivialTypeSourceInfo(UnsignedInt128Ty);
|
|
|
|
UInt128Decl = TypedefDecl::Create(const_cast<ASTContext &>(*this),
|
|
|
|
getTranslationUnitDecl(),
|
|
|
|
SourceLocation(),
|
|
|
|
SourceLocation(),
|
|
|
|
&Idents.get("__uint128_t"),
|
|
|
|
TInfo);
|
|
|
|
}
|
|
|
|
|
|
|
|
return UInt128Decl;
|
|
|
|
}
|
2007-07-11 21:01:13 +04:00
|
|
|
|
2009-10-24 03:03:21 +04:00
|
|
|
void ASTContext::InitBuiltinType(CanQualType &R, BuiltinType::Kind K) {
|
2009-09-25 03:30:46 +04:00
|
|
|
BuiltinType *Ty = new (*this, TypeAlignment) BuiltinType(K);
|
2009-10-24 03:03:21 +04:00
|
|
|
R = CanQualType::CreateUnsafe(QualType(Ty, 0));
|
2009-09-25 03:30:46 +04:00
|
|
|
Types.push_back(Ty);
|
2007-07-11 21:01:13 +04:00
|
|
|
}
|
|
|
|
|
2011-09-02 04:18:52 +04:00
|
|
|
void ASTContext::InitBuiltinTypes(const TargetInfo &Target) {
|
|
|
|
assert((!this->Target || this->Target == &Target) &&
|
|
|
|
"Incorrect target reinitialization");
|
2007-07-11 21:01:13 +04:00
|
|
|
assert(VoidTy.isNull() && "Context reinitialized?");
|
2009-09-09 19:08:12 +04:00
|
|
|
|
2011-09-02 04:18:52 +04:00
|
|
|
this->Target = &Target;
|
|
|
|
|
|
|
|
ABI.reset(createCXXABI(Target));
|
|
|
|
AddrSpaceMap = getAddressSpaceMap(Target, LangOpts);
|
|
|
|
|
2007-07-11 21:01:13 +04:00
|
|
|
// C99 6.2.5p19.
|
|
|
|
InitBuiltinType(VoidTy, BuiltinType::Void);
|
2009-09-09 19:08:12 +04:00
|
|
|
|
2007-07-11 21:01:13 +04:00
|
|
|
// C99 6.2.5p2.
|
|
|
|
InitBuiltinType(BoolTy, BuiltinType::Bool);
|
|
|
|
// C99 6.2.5p3.
|
2009-06-05 11:05:05 +04:00
|
|
|
if (LangOpts.CharIsSigned)
|
2007-07-11 21:01:13 +04:00
|
|
|
InitBuiltinType(CharTy, BuiltinType::Char_S);
|
|
|
|
else
|
|
|
|
InitBuiltinType(CharTy, BuiltinType::Char_U);
|
|
|
|
// C99 6.2.5p4.
|
|
|
|
InitBuiltinType(SignedCharTy, BuiltinType::SChar);
|
|
|
|
InitBuiltinType(ShortTy, BuiltinType::Short);
|
|
|
|
InitBuiltinType(IntTy, BuiltinType::Int);
|
|
|
|
InitBuiltinType(LongTy, BuiltinType::Long);
|
|
|
|
InitBuiltinType(LongLongTy, BuiltinType::LongLong);
|
2009-09-09 19:08:12 +04:00
|
|
|
|
2007-07-11 21:01:13 +04:00
|
|
|
// C99 6.2.5p6.
|
|
|
|
InitBuiltinType(UnsignedCharTy, BuiltinType::UChar);
|
|
|
|
InitBuiltinType(UnsignedShortTy, BuiltinType::UShort);
|
|
|
|
InitBuiltinType(UnsignedIntTy, BuiltinType::UInt);
|
|
|
|
InitBuiltinType(UnsignedLongTy, BuiltinType::ULong);
|
|
|
|
InitBuiltinType(UnsignedLongLongTy, BuiltinType::ULongLong);
|
2009-09-09 19:08:12 +04:00
|
|
|
|
2007-07-11 21:01:13 +04:00
|
|
|
// C99 6.2.5p10.
|
|
|
|
InitBuiltinType(FloatTy, BuiltinType::Float);
|
|
|
|
InitBuiltinType(DoubleTy, BuiltinType::Double);
|
|
|
|
InitBuiltinType(LongDoubleTy, BuiltinType::LongDouble);
|
2008-08-09 20:51:54 +04:00
|
|
|
|
2009-04-30 06:43:43 +04:00
|
|
|
// GNU extension, 128-bit integers.
|
|
|
|
InitBuiltinType(Int128Ty, BuiltinType::Int128);
|
|
|
|
InitBuiltinType(UnsignedInt128Ty, BuiltinType::UInt128);
|
|
|
|
|
2010-12-26 02:25:43 +03:00
|
|
|
if (LangOpts.CPlusPlus) { // C++ 3.9.1p5
|
2011-04-30 23:24:24 +04:00
|
|
|
if (TargetInfo::isTypeSigned(Target.getWCharType()))
|
2010-12-26 02:25:43 +03:00
|
|
|
InitBuiltinType(WCharTy, BuiltinType::WChar_S);
|
|
|
|
else // -fshort-wchar makes wchar_t be unsigned.
|
|
|
|
InitBuiltinType(WCharTy, BuiltinType::WChar_U);
|
|
|
|
} else // C99
|
2009-02-27 02:43:47 +03:00
|
|
|
WCharTy = getFromTargetType(Target.getWCharType());
|
2008-08-09 20:51:54 +04:00
|
|
|
|
2009-07-14 10:30:34 +04:00
|
|
|
if (LangOpts.CPlusPlus) // C++0x 3.9.1p5, extension for C++
|
|
|
|
InitBuiltinType(Char16Ty, BuiltinType::Char16);
|
|
|
|
else // C99
|
|
|
|
Char16Ty = getFromTargetType(Target.getChar16Type());
|
|
|
|
|
|
|
|
if (LangOpts.CPlusPlus) // C++0x 3.9.1p5, extension for C++
|
|
|
|
InitBuiltinType(Char32Ty, BuiltinType::Char32);
|
|
|
|
else // C99
|
|
|
|
Char32Ty = getFromTargetType(Target.getChar32Type());
|
|
|
|
|
2008-12-06 02:32:09 +03:00
|
|
|
// Placeholder type for type-dependent expressions whose type is
|
|
|
|
// completely unknown. No code should ever check a type against
|
|
|
|
// DependentTy and users should never see it; however, it is here to
|
|
|
|
// help diagnose failures to properly check for type-dependent
|
|
|
|
// expressions.
|
|
|
|
InitBuiltinType(DependentTy, BuiltinType::Dependent);
|
2008-10-21 20:13:35 +04:00
|
|
|
|
2010-10-12 04:20:44 +04:00
|
|
|
// Placeholder type for functions.
|
|
|
|
InitBuiltinType(OverloadTy, BuiltinType::Overload);
|
|
|
|
|
2011-04-27 00:42:42 +04:00
|
|
|
// Placeholder type for bound members.
|
|
|
|
InitBuiltinType(BoundMemberTy, BuiltinType::BoundMember);
|
|
|
|
|
2011-10-25 21:37:35 +04:00
|
|
|
// Placeholder type for pseudo-objects.
|
|
|
|
InitBuiltinType(PseudoObjectTy, BuiltinType::PseudoObject);
|
|
|
|
|
2011-04-07 12:22:57 +04:00
|
|
|
// "any" type; useful for debugger-like clients.
|
|
|
|
InitBuiltinType(UnknownAnyTy, BuiltinType::UnknownAny);
|
|
|
|
|
2011-10-17 22:09:15 +04:00
|
|
|
// Placeholder type for unbridged ARC casts.
|
|
|
|
InitBuiltinType(ARCUnbridgedCastTy, BuiltinType::ARCUnbridgedCast);
|
|
|
|
|
2007-07-11 21:01:13 +04:00
|
|
|
// C99 6.2.5p11.
|
|
|
|
FloatComplexTy = getComplexType(FloatTy);
|
|
|
|
DoubleComplexTy = getComplexType(DoubleTy);
|
|
|
|
LongDoubleComplexTy = getComplexType(LongDoubleTy);
|
2008-10-21 20:13:35 +04:00
|
|
|
|
2007-10-15 18:41:52 +04:00
|
|
|
BuiltinVaListType = QualType();
|
2009-09-09 19:08:12 +04:00
|
|
|
|
2009-11-21 22:53:08 +03:00
|
|
|
// Builtin types for 'id', 'Class', and 'SEL'.
|
2009-07-15 22:40:39 +04:00
|
|
|
InitBuiltinType(ObjCBuiltinIdTy, BuiltinType::ObjCId);
|
|
|
|
InitBuiltinType(ObjCBuiltinClassTy, BuiltinType::ObjCClass);
|
2009-11-21 22:53:08 +03:00
|
|
|
InitBuiltinType(ObjCBuiltinSelTy, BuiltinType::ObjCSel);
|
2012-03-07 00:05:56 +04:00
|
|
|
|
|
|
|
// Builtin type for __objc_yes and __objc_no
|
|
|
|
ObjCBuiltinBoolTy = SignedCharTy;
|
|
|
|
|
2008-01-07 22:49:32 +03:00
|
|
|
ObjCConstantStringType = QualType();
|
2009-09-09 19:08:12 +04:00
|
|
|
|
2007-10-30 01:57:28 +03:00
|
|
|
// void * type
|
|
|
|
VoidPtrTy = getPointerType(VoidTy);
|
2009-05-10 22:38:11 +04:00
|
|
|
|
|
|
|
// nullptr type (C++0x 2.14.7)
|
|
|
|
InitBuiltinType(NullPtrTy, BuiltinType::NullPtr);
|
2011-10-15 03:23:15 +04:00
|
|
|
|
|
|
|
// half type (OpenCL 6.1.1.1) / ARM NEON __fp16
|
|
|
|
InitBuiltinType(HalfTy, BuiltinType::Half);
|
2007-07-11 21:01:13 +04:00
|
|
|
}
|
|
|
|
|
2011-09-26 03:23:43 +04:00
|
|
|
DiagnosticsEngine &ASTContext::getDiagnostics() const {
|
2010-09-22 18:32:24 +04:00
|
|
|
return SourceMgr.getDiagnostics();
|
|
|
|
}
|
|
|
|
|
2010-08-30 20:49:28 +04:00
|
|
|
AttrVec& ASTContext::getDeclAttrs(const Decl *D) {
|
|
|
|
AttrVec *&Result = DeclAttrs[D];
|
|
|
|
if (!Result) {
|
|
|
|
void *Mem = Allocate(sizeof(AttrVec));
|
|
|
|
Result = new (Mem) AttrVec;
|
|
|
|
}
|
|
|
|
|
|
|
|
return *Result;
|
|
|
|
}
|
|
|
|
|
|
|
|
/// \brief Erase the attributes corresponding to the given declaration.
|
|
|
|
void ASTContext::eraseDeclAttrs(const Decl *D) {
|
|
|
|
llvm::DenseMap<const Decl*, AttrVec*>::iterator Pos = DeclAttrs.find(D);
|
|
|
|
if (Pos != DeclAttrs.end()) {
|
|
|
|
Pos->second->~AttrVec();
|
|
|
|
DeclAttrs.erase(Pos);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2009-10-08 11:24:58 +04:00
|
|
|
MemberSpecializationInfo *
|
2009-10-15 00:14:33 +04:00
|
|
|
ASTContext::getInstantiatedFromStaticDataMember(const VarDecl *Var) {
|
2009-07-25 00:34:43 +04:00
|
|
|
assert(Var->isStaticDataMember() && "Not a static data member");
|
2009-10-15 00:14:33 +04:00
|
|
|
llvm::DenseMap<const VarDecl *, MemberSpecializationInfo *>::iterator Pos
|
2009-07-25 00:34:43 +04:00
|
|
|
= InstantiatedFromStaticDataMember.find(Var);
|
|
|
|
if (Pos == InstantiatedFromStaticDataMember.end())
|
|
|
|
return 0;
|
2009-09-09 19:08:12 +04:00
|
|
|
|
2009-07-25 00:34:43 +04:00
|
|
|
return Pos->second;
|
|
|
|
}
|
|
|
|
|
2009-09-09 19:08:12 +04:00
|
|
|
void
|
2009-10-08 11:24:58 +04:00
|
|
|
ASTContext::setInstantiatedFromStaticDataMember(VarDecl *Inst, VarDecl *Tmpl,
|
2010-07-05 01:44:00 +04:00
|
|
|
TemplateSpecializationKind TSK,
|
|
|
|
SourceLocation PointOfInstantiation) {
|
2009-07-25 00:34:43 +04:00
|
|
|
assert(Inst->isStaticDataMember() && "Not a static data member");
|
|
|
|
assert(Tmpl->isStaticDataMember() && "Not a static data member");
|
|
|
|
assert(!InstantiatedFromStaticDataMember[Inst] &&
|
|
|
|
"Already noted what static data member was instantiated from");
|
2009-10-08 11:24:58 +04:00
|
|
|
InstantiatedFromStaticDataMember[Inst]
|
2010-07-05 01:44:00 +04:00
|
|
|
= new (*this) MemberSpecializationInfo(Tmpl, TSK, PointOfInstantiation);
|
2009-07-25 00:34:43 +04:00
|
|
|
}
|
|
|
|
|
2011-08-14 07:52:19 +04:00
|
|
|
FunctionDecl *ASTContext::getClassScopeSpecializationPattern(
|
|
|
|
const FunctionDecl *FD){
|
|
|
|
assert(FD && "Specialization is 0");
|
|
|
|
llvm::DenseMap<const FunctionDecl*, FunctionDecl *>::const_iterator Pos
|
2011-08-14 18:28:49 +04:00
|
|
|
= ClassScopeSpecializationPattern.find(FD);
|
|
|
|
if (Pos == ClassScopeSpecializationPattern.end())
|
2011-08-14 07:52:19 +04:00
|
|
|
return 0;
|
|
|
|
|
|
|
|
return Pos->second;
|
|
|
|
}
|
|
|
|
|
|
|
|
void ASTContext::setClassScopeSpecializationPattern(FunctionDecl *FD,
|
|
|
|
FunctionDecl *Pattern) {
|
|
|
|
assert(FD && "Specialization is 0");
|
|
|
|
assert(Pattern && "Class scope specialization pattern is 0");
|
2011-08-14 18:28:49 +04:00
|
|
|
ClassScopeSpecializationPattern[FD] = Pattern;
|
2011-08-14 07:52:19 +04:00
|
|
|
}
|
|
|
|
|
2009-11-18 05:36:19 +03:00
|
|
|
NamedDecl *
|
2009-12-05 01:46:56 +03:00
|
|
|
ASTContext::getInstantiatedFromUsingDecl(UsingDecl *UUD) {
|
2009-11-18 05:36:19 +03:00
|
|
|
llvm::DenseMap<UsingDecl *, NamedDecl *>::const_iterator Pos
|
2009-12-05 01:46:56 +03:00
|
|
|
= InstantiatedFromUsingDecl.find(UUD);
|
|
|
|
if (Pos == InstantiatedFromUsingDecl.end())
|
2009-08-29 23:37:28 +04:00
|
|
|
return 0;
|
2009-09-09 19:08:12 +04:00
|
|
|
|
2009-08-29 23:37:28 +04:00
|
|
|
return Pos->second;
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
2009-12-05 01:46:56 +03:00
|
|
|
ASTContext::setInstantiatedFromUsingDecl(UsingDecl *Inst, NamedDecl *Pattern) {
|
|
|
|
assert((isa<UsingDecl>(Pattern) ||
|
|
|
|
isa<UnresolvedUsingValueDecl>(Pattern) ||
|
|
|
|
isa<UnresolvedUsingTypenameDecl>(Pattern)) &&
|
|
|
|
"pattern decl is not a using decl");
|
|
|
|
assert(!InstantiatedFromUsingDecl[Inst] && "pattern already exists");
|
|
|
|
InstantiatedFromUsingDecl[Inst] = Pattern;
|
|
|
|
}
|
|
|
|
|
|
|
|
UsingShadowDecl *
|
|
|
|
ASTContext::getInstantiatedFromUsingShadowDecl(UsingShadowDecl *Inst) {
|
|
|
|
llvm::DenseMap<UsingShadowDecl*, UsingShadowDecl*>::const_iterator Pos
|
|
|
|
= InstantiatedFromUsingShadowDecl.find(Inst);
|
|
|
|
if (Pos == InstantiatedFromUsingShadowDecl.end())
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
return Pos->second;
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
ASTContext::setInstantiatedFromUsingShadowDecl(UsingShadowDecl *Inst,
|
|
|
|
UsingShadowDecl *Pattern) {
|
|
|
|
assert(!InstantiatedFromUsingShadowDecl[Inst] && "pattern already exists");
|
|
|
|
InstantiatedFromUsingShadowDecl[Inst] = Pattern;
|
2009-08-29 23:37:28 +04:00
|
|
|
}
|
|
|
|
|
2009-09-01 08:26:58 +04:00
|
|
|
FieldDecl *ASTContext::getInstantiatedFromUnnamedFieldDecl(FieldDecl *Field) {
|
|
|
|
llvm::DenseMap<FieldDecl *, FieldDecl *>::iterator Pos
|
|
|
|
= InstantiatedFromUnnamedFieldDecl.find(Field);
|
|
|
|
if (Pos == InstantiatedFromUnnamedFieldDecl.end())
|
|
|
|
return 0;
|
2009-09-09 19:08:12 +04:00
|
|
|
|
2009-09-01 08:26:58 +04:00
|
|
|
return Pos->second;
|
|
|
|
}
|
|
|
|
|
|
|
|
void ASTContext::setInstantiatedFromUnnamedFieldDecl(FieldDecl *Inst,
|
|
|
|
FieldDecl *Tmpl) {
|
|
|
|
assert(!Inst->getDeclName() && "Instantiated field decl is not unnamed");
|
|
|
|
assert(!Tmpl->getDeclName() && "Template field decl is not unnamed");
|
|
|
|
assert(!InstantiatedFromUnnamedFieldDecl[Inst] &&
|
|
|
|
"Already noted what unnamed field was instantiated from");
|
2009-09-09 19:08:12 +04:00
|
|
|
|
2009-09-01 08:26:58 +04:00
|
|
|
InstantiatedFromUnnamedFieldDecl[Inst] = Tmpl;
|
|
|
|
}
|
|
|
|
|
2011-04-27 21:14:21 +04:00
|
|
|
bool ASTContext::ZeroBitfieldFollowsNonBitfield(const FieldDecl *FD,
|
|
|
|
const FieldDecl *LastFD) const {
|
|
|
|
return (FD->isBitField() && LastFD && !LastFD->isBitField() &&
|
2011-10-10 22:28:20 +04:00
|
|
|
FD->getBitWidthValue(*this) == 0);
|
2011-04-27 21:14:21 +04:00
|
|
|
}
|
|
|
|
|
2011-05-02 21:20:56 +04:00
|
|
|
bool ASTContext::ZeroBitfieldFollowsBitfield(const FieldDecl *FD,
|
|
|
|
const FieldDecl *LastFD) const {
|
|
|
|
return (FD->isBitField() && LastFD && LastFD->isBitField() &&
|
2011-10-10 22:28:20 +04:00
|
|
|
FD->getBitWidthValue(*this) == 0 &&
|
|
|
|
LastFD->getBitWidthValue(*this) != 0);
|
2011-05-02 21:20:56 +04:00
|
|
|
}
|
|
|
|
|
2011-05-04 22:51:37 +04:00
|
|
|
bool ASTContext::BitfieldFollowsBitfield(const FieldDecl *FD,
|
|
|
|
const FieldDecl *LastFD) const {
|
|
|
|
return (FD->isBitField() && LastFD && LastFD->isBitField() &&
|
2011-10-10 22:28:20 +04:00
|
|
|
FD->getBitWidthValue(*this) &&
|
|
|
|
LastFD->getBitWidthValue(*this));
|
2011-05-04 22:51:37 +04:00
|
|
|
}
|
|
|
|
|
2011-08-05 03:34:15 +04:00
|
|
|
bool ASTContext::NonBitfieldFollowsBitfield(const FieldDecl *FD,
|
2011-05-07 01:56:12 +04:00
|
|
|
const FieldDecl *LastFD) const {
|
|
|
|
return (!FD->isBitField() && LastFD && LastFD->isBitField() &&
|
2011-10-10 22:28:20 +04:00
|
|
|
LastFD->getBitWidthValue(*this));
|
2011-05-07 01:56:12 +04:00
|
|
|
}
|
|
|
|
|
2011-08-05 03:34:15 +04:00
|
|
|
bool ASTContext::BitfieldFollowsNonBitfield(const FieldDecl *FD,
|
2011-05-07 01:56:12 +04:00
|
|
|
const FieldDecl *LastFD) const {
|
|
|
|
return (FD->isBitField() && LastFD && !LastFD->isBitField() &&
|
2011-10-10 22:28:20 +04:00
|
|
|
FD->getBitWidthValue(*this));
|
2011-05-07 01:56:12 +04:00
|
|
|
}
|
|
|
|
|
2010-03-03 02:58:15 +03:00
|
|
|
ASTContext::overridden_cxx_method_iterator
|
|
|
|
ASTContext::overridden_methods_begin(const CXXMethodDecl *Method) const {
|
|
|
|
llvm::DenseMap<const CXXMethodDecl *, CXXMethodVector>::const_iterator Pos
|
|
|
|
= OverriddenMethods.find(Method);
|
|
|
|
if (Pos == OverriddenMethods.end())
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
return Pos->second.begin();
|
|
|
|
}
|
|
|
|
|
|
|
|
ASTContext::overridden_cxx_method_iterator
|
|
|
|
ASTContext::overridden_methods_end(const CXXMethodDecl *Method) const {
|
|
|
|
llvm::DenseMap<const CXXMethodDecl *, CXXMethodVector>::const_iterator Pos
|
|
|
|
= OverriddenMethods.find(Method);
|
|
|
|
if (Pos == OverriddenMethods.end())
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
return Pos->second.end();
|
|
|
|
}
|
|
|
|
|
2010-07-05 01:44:35 +04:00
|
|
|
unsigned
|
|
|
|
ASTContext::overridden_methods_size(const CXXMethodDecl *Method) const {
|
|
|
|
llvm::DenseMap<const CXXMethodDecl *, CXXMethodVector>::const_iterator Pos
|
|
|
|
= OverriddenMethods.find(Method);
|
|
|
|
if (Pos == OverriddenMethods.end())
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
return Pos->second.size();
|
|
|
|
}
|
|
|
|
|
2010-03-03 02:58:15 +03:00
|
|
|
void ASTContext::addOverriddenMethod(const CXXMethodDecl *Method,
|
|
|
|
const CXXMethodDecl *Overridden) {
|
|
|
|
OverriddenMethods[Method].push_back(Overridden);
|
|
|
|
}
|
|
|
|
|
2011-12-03 04:30:27 +04:00
|
|
|
void ASTContext::addedLocalImportDecl(ImportDecl *Import) {
|
|
|
|
assert(!Import->NextLocalImport && "Import declaration already in the chain");
|
|
|
|
assert(!Import->isFromASTFile() && "Non-local import declaration");
|
|
|
|
if (!FirstLocalImport) {
|
|
|
|
FirstLocalImport = Import;
|
|
|
|
LastLocalImport = Import;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
LastLocalImport->NextLocalImport = Import;
|
|
|
|
LastLocalImport = Import;
|
|
|
|
}
|
|
|
|
|
2007-07-18 21:52:12 +04:00
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
// Type Sizing and Analysis
|
|
|
|
//===----------------------------------------------------------------------===//
|
2007-07-14 02:13:22 +04:00
|
|
|
|
2008-06-30 22:32:54 +04:00
|
|
|
/// getFloatTypeSemantics - Return the APFloat 'semantics' for the specified
|
|
|
|
/// scalar floating point type.
|
|
|
|
const llvm::fltSemantics &ASTContext::getFloatTypeSemantics(QualType T) const {
|
2009-09-22 03:43:11 +04:00
|
|
|
const BuiltinType *BT = T->getAs<BuiltinType>();
|
2008-06-30 22:32:54 +04:00
|
|
|
assert(BT && "Not a floating point type!");
|
|
|
|
switch (BT->getKind()) {
|
2011-09-23 09:06:16 +04:00
|
|
|
default: llvm_unreachable("Not a floating point type!");
|
2011-10-15 03:23:15 +04:00
|
|
|
case BuiltinType::Half: return Target->getHalfFormat();
|
2011-09-02 04:18:52 +04:00
|
|
|
case BuiltinType::Float: return Target->getFloatFormat();
|
|
|
|
case BuiltinType::Double: return Target->getDoubleFormat();
|
|
|
|
case BuiltinType::LongDouble: return Target->getLongDoubleFormat();
|
2008-06-30 22:32:54 +04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2010-01-27 20:10:57 +03:00
|
|
|
/// getDeclAlign - Return a conservative estimate of the alignment of the
|
2009-01-25 00:53:27 +03:00
|
|
|
/// specified decl. Note that bitfields do not have a valid alignment, so
|
|
|
|
/// this method will assert on them.
|
2009-11-23 20:18:46 +03:00
|
|
|
/// If @p RefAsPointee, references are treated like their underlying type
|
|
|
|
/// (for alignof), else they're treated like pointers (for CodeGen).
|
2011-01-12 12:06:06 +03:00
|
|
|
CharUnits ASTContext::getDeclAlign(const Decl *D, bool RefAsPointee) const {
|
2011-09-02 04:18:52 +04:00
|
|
|
unsigned Align = Target->getCharWidth();
|
2009-02-22 05:56:25 +03:00
|
|
|
|
2010-10-08 22:24:19 +04:00
|
|
|
bool UseAlignAttrOnly = false;
|
|
|
|
if (unsigned AlignFromAttr = D->getMaxAlignment()) {
|
|
|
|
Align = AlignFromAttr;
|
|
|
|
|
|
|
|
// __attribute__((aligned)) can increase or decrease alignment
|
|
|
|
// *except* on a struct or struct member, where it only increases
|
|
|
|
// alignment unless 'packed' is also specified.
|
|
|
|
//
|
2011-09-29 22:04:28 +04:00
|
|
|
// It is an error for alignas to decrease alignment, so we can
|
2010-10-08 22:24:19 +04:00
|
|
|
// ignore that possibility; Sema should diagnose it.
|
|
|
|
if (isa<FieldDecl>(D)) {
|
|
|
|
UseAlignAttrOnly = D->hasAttr<PackedAttr>() ||
|
|
|
|
cast<FieldDecl>(D)->getParent()->hasAttr<PackedAttr>();
|
|
|
|
} else {
|
|
|
|
UseAlignAttrOnly = true;
|
|
|
|
}
|
|
|
|
}
|
2011-05-06 01:19:14 +04:00
|
|
|
else if (isa<FieldDecl>(D))
|
|
|
|
UseAlignAttrOnly =
|
|
|
|
D->hasAttr<PackedAttr>() ||
|
|
|
|
cast<FieldDecl>(D)->getParent()->hasAttr<PackedAttr>();
|
2009-02-22 05:56:25 +03:00
|
|
|
|
2011-01-20 10:57:12 +03:00
|
|
|
// If we're using the align attribute only, just ignore everything
|
|
|
|
// else about the declaration and its type.
|
2010-10-08 22:24:19 +04:00
|
|
|
if (UseAlignAttrOnly) {
|
2011-01-20 10:57:12 +03:00
|
|
|
// do nothing
|
|
|
|
|
2010-10-08 22:24:19 +04:00
|
|
|
} else if (const ValueDecl *VD = dyn_cast<ValueDecl>(D)) {
|
2009-01-25 00:53:27 +03:00
|
|
|
QualType T = VD->getType();
|
2009-07-30 01:53:49 +04:00
|
|
|
if (const ReferenceType* RT = T->getAs<ReferenceType>()) {
|
2009-11-23 20:18:46 +03:00
|
|
|
if (RefAsPointee)
|
|
|
|
T = RT->getPointeeType();
|
|
|
|
else
|
|
|
|
T = getPointerType(RT->getPointeeType());
|
|
|
|
}
|
|
|
|
if (!T->isIncompleteType() && !T->isFunctionType()) {
|
2011-01-19 13:06:00 +03:00
|
|
|
// Adjust alignments of declarations with array type by the
|
|
|
|
// large-array alignment on the target.
|
2011-09-02 04:18:52 +04:00
|
|
|
unsigned MinWidth = Target->getLargeArrayMinWidth();
|
2011-01-19 13:06:00 +03:00
|
|
|
const ArrayType *arrayType;
|
|
|
|
if (MinWidth && (arrayType = getAsArrayType(T))) {
|
|
|
|
if (isa<VariableArrayType>(arrayType))
|
2011-09-02 04:18:52 +04:00
|
|
|
Align = std::max(Align, Target->getLargeArrayAlign());
|
2011-01-19 13:06:00 +03:00
|
|
|
else if (isa<ConstantArrayType>(arrayType) &&
|
|
|
|
MinWidth <= getTypeSize(cast<ConstantArrayType>(arrayType)))
|
2011-09-02 04:18:52 +04:00
|
|
|
Align = std::max(Align, Target->getLargeArrayAlign());
|
2011-01-19 13:06:00 +03:00
|
|
|
|
|
|
|
// Walk through any array types while we're at it.
|
|
|
|
T = getBaseElementType(arrayType);
|
2010-06-05 03:15:27 +04:00
|
|
|
}
|
2011-07-26 11:03:04 +04:00
|
|
|
Align = std::max(Align, getPreferredTypeAlign(T.getTypePtr()));
|
2009-02-22 05:56:25 +03:00
|
|
|
}
|
2011-01-20 10:57:12 +03:00
|
|
|
|
|
|
|
// Fields can be subject to extra alignment constraints, like if
|
|
|
|
// the field is packed, the struct is packed, or the struct has a
|
|
|
|
// a max-field-alignment constraint (#pragma pack). So calculate
|
|
|
|
// the actual alignment of the field within the struct, and then
|
|
|
|
// (as we're expected to) constrain that by the alignment of the type.
|
|
|
|
if (const FieldDecl *field = dyn_cast<FieldDecl>(VD)) {
|
|
|
|
// So calculate the alignment of the field.
|
|
|
|
const ASTRecordLayout &layout = getASTRecordLayout(field->getParent());
|
|
|
|
|
|
|
|
// Start with the record's overall alignment.
|
2011-02-15 05:32:40 +03:00
|
|
|
unsigned fieldAlign = toBits(layout.getAlignment());
|
2011-01-20 10:57:12 +03:00
|
|
|
|
|
|
|
// Use the GCD of that and the offset within the record.
|
|
|
|
uint64_t offset = layout.getFieldOffset(field->getFieldIndex());
|
|
|
|
if (offset > 0) {
|
|
|
|
// Alignment is always a power of 2, so the GCD will be a power of 2,
|
|
|
|
// which means we get to do this crazy thing instead of Euclid's.
|
|
|
|
uint64_t lowBitOfOffset = offset & (~offset + 1);
|
|
|
|
if (lowBitOfOffset < fieldAlign)
|
|
|
|
fieldAlign = static_cast<unsigned>(lowBitOfOffset);
|
|
|
|
}
|
|
|
|
|
|
|
|
Align = std::min(Align, fieldAlign);
|
2010-02-23 07:52:00 +03:00
|
|
|
}
|
2009-01-25 00:53:27 +03:00
|
|
|
}
|
2009-02-22 05:56:25 +03:00
|
|
|
|
2011-01-15 21:38:59 +03:00
|
|
|
return toCharUnitsFromBits(Align);
|
2009-01-25 00:53:27 +03:00
|
|
|
}
|
2008-06-30 22:32:54 +04:00
|
|
|
|
2010-05-20 05:18:31 +04:00
|
|
|
std::pair<CharUnits, CharUnits>
|
2011-02-20 04:55:18 +03:00
|
|
|
ASTContext::getTypeInfoInChars(const Type *T) const {
|
2010-05-20 05:18:31 +04:00
|
|
|
std::pair<uint64_t, unsigned> Info = getTypeInfo(T);
|
2011-01-15 21:38:59 +03:00
|
|
|
return std::make_pair(toCharUnitsFromBits(Info.first),
|
|
|
|
toCharUnitsFromBits(Info.second));
|
2010-05-20 05:18:31 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
std::pair<CharUnits, CharUnits>
|
2011-02-20 04:55:18 +03:00
|
|
|
ASTContext::getTypeInfoInChars(QualType T) const {
|
2010-05-20 05:18:31 +04:00
|
|
|
return getTypeInfoInChars(T.getTypePtr());
|
|
|
|
}
|
|
|
|
|
2012-03-09 08:12:54 +04:00
|
|
|
std::pair<uint64_t, unsigned> ASTContext::getTypeInfo(const Type *T) const {
|
|
|
|
TypeInfoMap::iterator it = MemoizedTypeInfo.find(T);
|
|
|
|
if (it != MemoizedTypeInfo.end())
|
|
|
|
return it->second;
|
|
|
|
|
|
|
|
std::pair<uint64_t, unsigned> Info = getTypeInfoImpl(T);
|
|
|
|
MemoizedTypeInfo.insert(std::make_pair(T, Info));
|
|
|
|
return Info;
|
|
|
|
}
|
|
|
|
|
|
|
|
/// getTypeInfoImpl - Return the size of the specified type, in bits. This
|
|
|
|
/// method does not work on incomplete types.
|
2009-09-24 23:53:00 +04:00
|
|
|
///
|
|
|
|
/// FIXME: Pointers into different addr spaces could have different sizes and
|
|
|
|
/// alignment requirements: getPointerInfo should take an AddrSpace, this
|
|
|
|
/// should take a QualType, &c.
|
2007-07-14 05:29:45 +04:00
|
|
|
std::pair<uint64_t, unsigned>
|
2012-03-09 08:12:54 +04:00
|
|
|
ASTContext::getTypeInfoImpl(const Type *T) const {
|
2009-02-27 21:32:39 +03:00
|
|
|
uint64_t Width=0;
|
|
|
|
unsigned Align=8;
|
2007-07-14 02:13:22 +04:00
|
|
|
switch (T->getTypeClass()) {
|
2009-02-27 02:50:07 +03:00
|
|
|
#define TYPE(Class, Base)
|
|
|
|
#define ABSTRACT_TYPE(Class, Base)
|
2009-04-30 21:32:17 +04:00
|
|
|
#define NON_CANONICAL_TYPE(Class, Base)
|
2009-02-27 02:50:07 +03:00
|
|
|
#define DEPENDENT_TYPE(Class, Base) case Type::Class:
|
|
|
|
#include "clang/AST/TypeNodes.def"
|
2011-06-28 20:49:23 +04:00
|
|
|
llvm_unreachable("Should not see dependent types");
|
2009-02-27 02:50:07 +03:00
|
|
|
|
2007-07-18 22:26:58 +04:00
|
|
|
case Type::FunctionNoProto:
|
|
|
|
case Type::FunctionProto:
|
2009-04-30 21:32:17 +04:00
|
|
|
// GCC extension: alignof(function) = 32 bits
|
|
|
|
Width = 0;
|
|
|
|
Align = 32;
|
|
|
|
break;
|
|
|
|
|
2009-02-27 02:50:07 +03:00
|
|
|
case Type::IncompleteArray:
|
2007-08-30 05:06:46 +04:00
|
|
|
case Type::VariableArray:
|
2009-04-30 21:32:17 +04:00
|
|
|
Width = 0;
|
|
|
|
Align = getTypeAlign(cast<ArrayType>(T)->getElementType());
|
|
|
|
break;
|
|
|
|
|
2007-08-30 05:06:46 +04:00
|
|
|
case Type::ConstantArray: {
|
2008-11-08 08:48:37 +03:00
|
|
|
const ConstantArrayType *CAT = cast<ConstantArrayType>(T);
|
2009-09-09 19:08:12 +04:00
|
|
|
|
2008-03-05 21:54:05 +03:00
|
|
|
std::pair<uint64_t, unsigned> EltInfo = getTypeInfo(CAT->getElementType());
|
2011-12-13 15:23:52 +04:00
|
|
|
uint64_t Size = CAT->getSize().getZExtValue();
|
2012-03-09 08:12:54 +04:00
|
|
|
assert((Size == 0 || EltInfo.first <= (uint64_t)(-1)/Size) &&
|
|
|
|
"Overflow in array type bit size evaluation");
|
2011-12-13 15:23:52 +04:00
|
|
|
Width = EltInfo.first*Size;
|
2007-07-20 02:06:24 +04:00
|
|
|
Align = EltInfo.second;
|
2011-04-27 01:05:39 +04:00
|
|
|
Width = llvm::RoundUpToAlignment(Width, Align);
|
2007-07-20 02:06:24 +04:00
|
|
|
break;
|
2007-12-29 08:10:55 +03:00
|
|
|
}
|
2008-04-19 03:10:10 +04:00
|
|
|
case Type::ExtVector:
|
2007-07-20 02:06:24 +04:00
|
|
|
case Type::Vector: {
|
2009-10-22 09:17:15 +04:00
|
|
|
const VectorType *VT = cast<VectorType>(T);
|
|
|
|
std::pair<uint64_t, unsigned> EltInfo = getTypeInfo(VT->getElementType());
|
|
|
|
Width = EltInfo.first*VT->getNumElements();
|
2008-05-30 13:31:38 +04:00
|
|
|
Align = Width;
|
2009-01-18 09:42:49 +03:00
|
|
|
// If the alignment is not a power of 2, round up to the next power of 2.
|
|
|
|
// This happens for non-power-of-2 length vectors.
|
2010-04-22 03:32:43 +04:00
|
|
|
if (Align & (Align-1)) {
|
2009-10-22 09:17:15 +04:00
|
|
|
Align = llvm::NextPowerOf2(Align);
|
|
|
|
Width = llvm::RoundUpToAlignment(Width, Align);
|
|
|
|
}
|
2007-07-20 02:06:24 +04:00
|
|
|
break;
|
|
|
|
}
|
2007-07-18 22:26:58 +04:00
|
|
|
|
2008-03-08 11:52:55 +03:00
|
|
|
case Type::Builtin:
|
2007-07-14 02:13:22 +04:00
|
|
|
switch (cast<BuiltinType>(T)->getKind()) {
|
2011-09-23 09:06:16 +04:00
|
|
|
default: llvm_unreachable("Unknown builtin type!");
|
2007-07-14 05:29:45 +04:00
|
|
|
case BuiltinType::Void:
|
2009-04-30 21:32:17 +04:00
|
|
|
// GCC extension: alignof(void) = 8 bits.
|
|
|
|
Width = 0;
|
|
|
|
Align = 8;
|
|
|
|
break;
|
|
|
|
|
2007-12-19 22:23:28 +03:00
|
|
|
case BuiltinType::Bool:
|
2011-09-02 04:18:52 +04:00
|
|
|
Width = Target->getBoolWidth();
|
|
|
|
Align = Target->getBoolAlign();
|
2007-12-19 22:23:28 +03:00
|
|
|
break;
|
2007-07-14 02:27:08 +04:00
|
|
|
case BuiltinType::Char_S:
|
|
|
|
case BuiltinType::Char_U:
|
|
|
|
case BuiltinType::UChar:
|
2007-12-19 22:23:28 +03:00
|
|
|
case BuiltinType::SChar:
|
2011-09-02 04:18:52 +04:00
|
|
|
Width = Target->getCharWidth();
|
|
|
|
Align = Target->getCharAlign();
|
2007-12-19 22:23:28 +03:00
|
|
|
break;
|
2010-12-26 02:25:43 +03:00
|
|
|
case BuiltinType::WChar_S:
|
|
|
|
case BuiltinType::WChar_U:
|
2011-09-02 04:18:52 +04:00
|
|
|
Width = Target->getWCharWidth();
|
|
|
|
Align = Target->getWCharAlign();
|
2008-08-09 20:51:54 +04:00
|
|
|
break;
|
2009-07-14 10:30:34 +04:00
|
|
|
case BuiltinType::Char16:
|
2011-09-02 04:18:52 +04:00
|
|
|
Width = Target->getChar16Width();
|
|
|
|
Align = Target->getChar16Align();
|
2009-07-14 10:30:34 +04:00
|
|
|
break;
|
|
|
|
case BuiltinType::Char32:
|
2011-09-02 04:18:52 +04:00
|
|
|
Width = Target->getChar32Width();
|
|
|
|
Align = Target->getChar32Align();
|
2009-07-14 10:30:34 +04:00
|
|
|
break;
|
2007-07-14 02:27:08 +04:00
|
|
|
case BuiltinType::UShort:
|
2007-12-19 22:23:28 +03:00
|
|
|
case BuiltinType::Short:
|
2011-09-02 04:18:52 +04:00
|
|
|
Width = Target->getShortWidth();
|
|
|
|
Align = Target->getShortAlign();
|
2007-12-19 22:23:28 +03:00
|
|
|
break;
|
2007-07-14 02:27:08 +04:00
|
|
|
case BuiltinType::UInt:
|
2007-12-19 22:23:28 +03:00
|
|
|
case BuiltinType::Int:
|
2011-09-02 04:18:52 +04:00
|
|
|
Width = Target->getIntWidth();
|
|
|
|
Align = Target->getIntAlign();
|
2007-12-19 22:23:28 +03:00
|
|
|
break;
|
2007-07-14 02:27:08 +04:00
|
|
|
case BuiltinType::ULong:
|
2007-12-19 22:23:28 +03:00
|
|
|
case BuiltinType::Long:
|
2011-09-02 04:18:52 +04:00
|
|
|
Width = Target->getLongWidth();
|
|
|
|
Align = Target->getLongAlign();
|
2007-12-19 22:23:28 +03:00
|
|
|
break;
|
2007-07-14 02:27:08 +04:00
|
|
|
case BuiltinType::ULongLong:
|
2007-12-19 22:23:28 +03:00
|
|
|
case BuiltinType::LongLong:
|
2011-09-02 04:18:52 +04:00
|
|
|
Width = Target->getLongLongWidth();
|
|
|
|
Align = Target->getLongLongAlign();
|
2007-12-19 22:23:28 +03:00
|
|
|
break;
|
2009-04-30 06:55:13 +04:00
|
|
|
case BuiltinType::Int128:
|
|
|
|
case BuiltinType::UInt128:
|
|
|
|
Width = 128;
|
|
|
|
Align = 128; // int128_t is 128-bit aligned on all targets.
|
|
|
|
break;
|
2011-10-15 03:23:15 +04:00
|
|
|
case BuiltinType::Half:
|
|
|
|
Width = Target->getHalfWidth();
|
|
|
|
Align = Target->getHalfAlign();
|
|
|
|
break;
|
2007-12-19 22:23:28 +03:00
|
|
|
case BuiltinType::Float:
|
2011-09-02 04:18:52 +04:00
|
|
|
Width = Target->getFloatWidth();
|
|
|
|
Align = Target->getFloatAlign();
|
2007-12-19 22:23:28 +03:00
|
|
|
break;
|
|
|
|
case BuiltinType::Double:
|
2011-09-02 04:18:52 +04:00
|
|
|
Width = Target->getDoubleWidth();
|
|
|
|
Align = Target->getDoubleAlign();
|
2007-12-19 22:23:28 +03:00
|
|
|
break;
|
|
|
|
case BuiltinType::LongDouble:
|
2011-09-02 04:18:52 +04:00
|
|
|
Width = Target->getLongDoubleWidth();
|
|
|
|
Align = Target->getLongDoubleAlign();
|
2007-12-19 22:23:28 +03:00
|
|
|
break;
|
2009-05-10 22:38:11 +04:00
|
|
|
case BuiltinType::NullPtr:
|
2011-09-02 04:18:52 +04:00
|
|
|
Width = Target->getPointerWidth(0); // C++ 3.9.1p11: sizeof(nullptr_t)
|
|
|
|
Align = Target->getPointerAlign(0); // == sizeof(void*)
|
2009-05-27 23:34:06 +04:00
|
|
|
break;
|
2010-08-02 22:03:20 +04:00
|
|
|
case BuiltinType::ObjCId:
|
|
|
|
case BuiltinType::ObjCClass:
|
|
|
|
case BuiltinType::ObjCSel:
|
2011-09-02 04:18:52 +04:00
|
|
|
Width = Target->getPointerWidth(0);
|
|
|
|
Align = Target->getPointerAlign(0);
|
2010-08-02 22:03:20 +04:00
|
|
|
break;
|
2007-07-14 02:13:22 +04:00
|
|
|
}
|
2007-07-16 03:46:53 +04:00
|
|
|
break;
|
2009-06-18 02:40:22 +04:00
|
|
|
case Type::ObjCObjectPointer:
|
2011-09-02 04:18:52 +04:00
|
|
|
Width = Target->getPointerWidth(0);
|
|
|
|
Align = Target->getPointerAlign(0);
|
2007-12-19 22:23:28 +03:00
|
|
|
break;
|
2008-09-24 19:05:44 +04:00
|
|
|
case Type::BlockPointer: {
|
2011-03-19 01:38:29 +03:00
|
|
|
unsigned AS = getTargetAddressSpace(
|
|
|
|
cast<BlockPointerType>(T)->getPointeeType());
|
2011-09-02 04:18:52 +04:00
|
|
|
Width = Target->getPointerWidth(AS);
|
|
|
|
Align = Target->getPointerAlign(AS);
|
2008-09-24 19:05:44 +04:00
|
|
|
break;
|
|
|
|
}
|
2009-11-23 20:18:46 +03:00
|
|
|
case Type::LValueReference:
|
|
|
|
case Type::RValueReference: {
|
|
|
|
// alignof and sizeof should never enter this code path here, so we go
|
|
|
|
// the pointer route.
|
2011-03-19 01:38:29 +03:00
|
|
|
unsigned AS = getTargetAddressSpace(
|
|
|
|
cast<ReferenceType>(T)->getPointeeType());
|
2011-09-02 04:18:52 +04:00
|
|
|
Width = Target->getPointerWidth(AS);
|
|
|
|
Align = Target->getPointerAlign(AS);
|
2009-11-23 20:18:46 +03:00
|
|
|
break;
|
|
|
|
}
|
2008-03-08 11:34:58 +03:00
|
|
|
case Type::Pointer: {
|
2011-03-19 01:38:29 +03:00
|
|
|
unsigned AS = getTargetAddressSpace(cast<PointerType>(T)->getPointeeType());
|
2011-09-02 04:18:52 +04:00
|
|
|
Width = Target->getPointerWidth(AS);
|
|
|
|
Align = Target->getPointerAlign(AS);
|
2008-03-08 11:34:58 +03:00
|
|
|
break;
|
|
|
|
}
|
2009-01-25 00:16:55 +03:00
|
|
|
case Type::MemberPointer: {
|
2010-08-16 07:33:14 +04:00
|
|
|
const MemberPointerType *MPT = cast<MemberPointerType>(T);
|
2009-09-09 19:08:12 +04:00
|
|
|
std::pair<uint64_t, unsigned> PtrDiffInfo =
|
2009-05-17 06:06:04 +04:00
|
|
|
getTypeInfo(getPointerDiffType());
|
2010-08-16 07:33:14 +04:00
|
|
|
Width = PtrDiffInfo.first * ABI->getMemberPointerSize(MPT);
|
2009-05-17 06:06:04 +04:00
|
|
|
Align = PtrDiffInfo.second;
|
|
|
|
break;
|
2009-01-25 00:16:55 +03:00
|
|
|
}
|
2007-07-18 22:26:58 +04:00
|
|
|
case Type::Complex: {
|
|
|
|
// Complex types have the same alignment as their elements, but twice the
|
|
|
|
// size.
|
2009-09-09 19:08:12 +04:00
|
|
|
std::pair<uint64_t, unsigned> EltInfo =
|
2008-03-05 21:54:05 +03:00
|
|
|
getTypeInfo(cast<ComplexType>(T)->getElementType());
|
2008-03-08 11:52:55 +03:00
|
|
|
Width = EltInfo.first*2;
|
2007-07-18 22:26:58 +04:00
|
|
|
Align = EltInfo.second;
|
|
|
|
break;
|
|
|
|
}
|
2010-05-15 15:32:37 +04:00
|
|
|
case Type::ObjCObject:
|
|
|
|
return getTypeInfo(cast<ObjCObjectType>(T)->getBaseType().getTypePtr());
|
2008-06-05 01:54:36 +04:00
|
|
|
case Type::ObjCInterface: {
|
2008-11-08 08:48:37 +03:00
|
|
|
const ObjCInterfaceType *ObjCI = cast<ObjCInterfaceType>(T);
|
2008-06-05 01:54:36 +04:00
|
|
|
const ASTRecordLayout &Layout = getASTObjCInterfaceLayout(ObjCI->getDecl());
|
2011-02-11 04:54:29 +03:00
|
|
|
Width = toBits(Layout.getSize());
|
2011-02-15 05:32:40 +03:00
|
|
|
Align = toBits(Layout.getAlignment());
|
2008-06-05 01:54:36 +04:00
|
|
|
break;
|
|
|
|
}
|
2009-02-27 02:50:07 +03:00
|
|
|
case Type::Record:
|
|
|
|
case Type::Enum: {
|
2008-11-08 08:48:37 +03:00
|
|
|
const TagType *TT = cast<TagType>(T);
|
|
|
|
|
|
|
|
if (TT->getDecl()->isInvalidDecl()) {
|
2011-04-20 21:29:44 +04:00
|
|
|
Width = 8;
|
|
|
|
Align = 8;
|
2008-08-10 01:35:13 +04:00
|
|
|
break;
|
|
|
|
}
|
2009-09-09 19:08:12 +04:00
|
|
|
|
2008-11-08 08:48:37 +03:00
|
|
|
if (const EnumType *ET = dyn_cast<EnumType>(TT))
|
2008-04-07 02:05:18 +04:00
|
|
|
return getTypeInfo(ET->getDecl()->getIntegerType());
|
|
|
|
|
2008-11-08 08:48:37 +03:00
|
|
|
const RecordType *RT = cast<RecordType>(TT);
|
2008-04-07 02:05:18 +04:00
|
|
|
const ASTRecordLayout &Layout = getASTRecordLayout(RT->getDecl());
|
2011-02-11 04:54:29 +03:00
|
|
|
Width = toBits(Layout.getSize());
|
2011-02-15 05:32:40 +03:00
|
|
|
Align = toBits(Layout.getAlignment());
|
2007-07-24 02:46:22 +04:00
|
|
|
break;
|
2007-07-14 02:13:22 +04:00
|
|
|
}
|
2009-03-31 02:58:21 +04:00
|
|
|
|
2009-10-22 09:17:15 +04:00
|
|
|
case Type::SubstTemplateTypeParm:
|
2009-10-18 13:09:24 +04:00
|
|
|
return getTypeInfo(cast<SubstTemplateTypeParmType>(T)->
|
|
|
|
getReplacementType().getTypePtr());
|
|
|
|
|
2011-02-20 06:19:35 +03:00
|
|
|
case Type::Auto: {
|
|
|
|
const AutoType *A = cast<AutoType>(T);
|
|
|
|
assert(A->isDeduced() && "Cannot request the size of a dependent type");
|
2011-02-22 23:00:16 +03:00
|
|
|
return getTypeInfo(A->getDeducedType().getTypePtr());
|
2011-02-20 06:19:35 +03:00
|
|
|
}
|
|
|
|
|
2010-12-10 19:29:40 +03:00
|
|
|
case Type::Paren:
|
|
|
|
return getTypeInfo(cast<ParenType>(T)->getInnerType().getTypePtr());
|
|
|
|
|
2009-04-30 21:32:17 +04:00
|
|
|
case Type::Typedef: {
|
2011-04-15 18:24:37 +04:00
|
|
|
const TypedefNameDecl *Typedef = cast<TypedefType>(T)->getDecl();
|
2010-08-27 04:11:28 +04:00
|
|
|
std::pair<uint64_t, unsigned> Info
|
|
|
|
= getTypeInfo(Typedef->getUnderlyingType().getTypePtr());
|
2011-02-20 01:55:41 +03:00
|
|
|
// If the typedef has an aligned attribute on it, it overrides any computed
|
|
|
|
// alignment we have. This violates the GCC documentation (which says that
|
|
|
|
// attribute(aligned) can only round up) but matches its implementation.
|
|
|
|
if (unsigned AttrAlign = Typedef->getMaxAlignment())
|
|
|
|
Align = AttrAlign;
|
|
|
|
else
|
|
|
|
Align = Info.second;
|
2010-08-27 04:11:28 +04:00
|
|
|
Width = Info.first;
|
2009-03-31 02:58:21 +04:00
|
|
|
break;
|
2008-04-07 02:05:18 +04:00
|
|
|
}
|
2009-04-30 21:32:17 +04:00
|
|
|
|
|
|
|
case Type::TypeOfExpr:
|
|
|
|
return getTypeInfo(cast<TypeOfExprType>(T)->getUnderlyingExpr()->getType()
|
|
|
|
.getTypePtr());
|
|
|
|
|
|
|
|
case Type::TypeOf:
|
|
|
|
return getTypeInfo(cast<TypeOfType>(T)->getUnderlyingType().getTypePtr());
|
|
|
|
|
2009-06-24 23:06:50 +04:00
|
|
|
case Type::Decltype:
|
|
|
|
return getTypeInfo(cast<DecltypeType>(T)->getUnderlyingExpr()->getType()
|
|
|
|
.getTypePtr());
|
|
|
|
|
2011-05-25 02:41:36 +04:00
|
|
|
case Type::UnaryTransform:
|
|
|
|
return getTypeInfo(cast<UnaryTransformType>(T)->getUnderlyingType());
|
|
|
|
|
2010-05-12 01:36:43 +04:00
|
|
|
case Type::Elaborated:
|
|
|
|
return getTypeInfo(cast<ElaboratedType>(T)->getNamedType().getTypePtr());
|
2009-09-09 19:08:12 +04:00
|
|
|
|
2011-01-06 04:58:22 +03:00
|
|
|
case Type::Attributed:
|
|
|
|
return getTypeInfo(
|
|
|
|
cast<AttributedType>(T)->getEquivalentType().getTypePtr());
|
|
|
|
|
2011-05-06 01:57:07 +04:00
|
|
|
case Type::TemplateSpecialization: {
|
2009-09-09 19:08:12 +04:00
|
|
|
assert(getCanonicalType(T) != T &&
|
2009-04-30 21:32:17 +04:00
|
|
|
"Cannot request the size of a dependent type");
|
2011-05-06 01:57:07 +04:00
|
|
|
const TemplateSpecializationType *TST = cast<TemplateSpecializationType>(T);
|
|
|
|
// A type alias template specialization may refer to a typedef with the
|
|
|
|
// aligned attribute on it.
|
|
|
|
if (TST->isTypeAlias())
|
|
|
|
return getTypeInfo(TST->getAliasedType().getTypePtr());
|
|
|
|
else
|
|
|
|
return getTypeInfo(getCanonicalType(T));
|
|
|
|
}
|
|
|
|
|
2011-10-07 03:00:33 +04:00
|
|
|
case Type::Atomic: {
|
2011-10-15 00:59:01 +04:00
|
|
|
std::pair<uint64_t, unsigned> Info
|
|
|
|
= getTypeInfo(cast<AtomicType>(T)->getValueType());
|
|
|
|
Width = Info.first;
|
|
|
|
Align = Info.second;
|
|
|
|
if (Width != 0 && Width <= Target->getMaxAtomicPromoteWidth() &&
|
|
|
|
llvm::isPowerOf2_64(Width)) {
|
|
|
|
// We can potentially perform lock-free atomic operations for this
|
|
|
|
// type; promote the alignment appropriately.
|
|
|
|
// FIXME: We could potentially promote the width here as well...
|
|
|
|
// is that worthwhile? (Non-struct atomic types generally have
|
|
|
|
// power-of-two size anyway, but structs might not. Requires a bit
|
|
|
|
// of implementation work to make sure we zero out the extra bits.)
|
|
|
|
Align = static_cast<unsigned>(Width);
|
|
|
|
}
|
2011-10-07 03:00:33 +04:00
|
|
|
}
|
|
|
|
|
2009-04-30 21:32:17 +04:00
|
|
|
}
|
2009-09-09 19:08:12 +04:00
|
|
|
|
2011-10-15 00:59:01 +04:00
|
|
|
assert(llvm::isPowerOf2_32(Align) && "Alignment must be power of 2");
|
2008-03-08 11:52:55 +03:00
|
|
|
return std::make_pair(Width, Align);
|
2007-07-14 02:13:22 +04:00
|
|
|
}
|
|
|
|
|
2011-01-15 21:38:59 +03:00
|
|
|
/// toCharUnitsFromBits - Convert a size in bits to a size in characters.
|
|
|
|
CharUnits ASTContext::toCharUnitsFromBits(int64_t BitSize) const {
|
|
|
|
return CharUnits::fromQuantity(BitSize / getCharWidth());
|
|
|
|
}
|
|
|
|
|
2011-02-11 04:54:29 +03:00
|
|
|
/// toBits - Convert a size in characters to a size in characters.
|
|
|
|
int64_t ASTContext::toBits(CharUnits CharSize) const {
|
|
|
|
return CharSize.getQuantity() * getCharWidth();
|
|
|
|
}
|
|
|
|
|
2009-12-22 17:23:30 +03:00
|
|
|
/// getTypeSizeInChars - Return the size of the specified type, in characters.
|
|
|
|
/// This method does not work on incomplete types.
|
2011-01-12 12:06:06 +03:00
|
|
|
CharUnits ASTContext::getTypeSizeInChars(QualType T) const {
|
2011-01-15 21:38:59 +03:00
|
|
|
return toCharUnitsFromBits(getTypeSize(T));
|
2009-12-22 17:23:30 +03:00
|
|
|
}
|
2011-01-12 12:06:06 +03:00
|
|
|
CharUnits ASTContext::getTypeSizeInChars(const Type *T) const {
|
2011-01-15 21:38:59 +03:00
|
|
|
return toCharUnitsFromBits(getTypeSize(T));
|
2009-12-22 17:23:30 +03:00
|
|
|
}
|
|
|
|
|
2010-01-26 20:25:18 +03:00
|
|
|
/// getTypeAlignInChars - Return the ABI-specified alignment of a type, in
|
2010-01-26 20:22:55 +03:00
|
|
|
/// characters. This method does not work on incomplete types.
|
2011-01-12 12:06:06 +03:00
|
|
|
CharUnits ASTContext::getTypeAlignInChars(QualType T) const {
|
2011-01-15 21:38:59 +03:00
|
|
|
return toCharUnitsFromBits(getTypeAlign(T));
|
2010-01-26 20:22:55 +03:00
|
|
|
}
|
2011-01-12 12:06:06 +03:00
|
|
|
CharUnits ASTContext::getTypeAlignInChars(const Type *T) const {
|
2011-01-15 21:38:59 +03:00
|
|
|
return toCharUnitsFromBits(getTypeAlign(T));
|
2010-01-26 20:22:55 +03:00
|
|
|
}
|
|
|
|
|
2009-01-27 21:08:34 +03:00
|
|
|
/// getPreferredTypeAlign - Return the "preferred" alignment of the specified
|
|
|
|
/// type for the current target in bits. This can be different than the ABI
|
|
|
|
/// alignment in cases where it is beneficial for performance to overalign
|
|
|
|
/// a data type.
|
2011-01-12 12:06:06 +03:00
|
|
|
unsigned ASTContext::getPreferredTypeAlign(const Type *T) const {
|
2009-01-27 21:08:34 +03:00
|
|
|
unsigned ABIAlign = getTypeAlign(T);
|
2009-05-26 01:27:19 +04:00
|
|
|
|
|
|
|
// Double and long long should be naturally aligned if possible.
|
2009-09-22 03:43:11 +04:00
|
|
|
if (const ComplexType* CT = T->getAs<ComplexType>())
|
2009-05-26 01:27:19 +04:00
|
|
|
T = CT->getElementType().getTypePtr();
|
|
|
|
if (T->isSpecificBuiltinType(BuiltinType::Double) ||
|
2012-03-22 00:20:47 +04:00
|
|
|
T->isSpecificBuiltinType(BuiltinType::LongLong) ||
|
|
|
|
T->isSpecificBuiltinType(BuiltinType::ULongLong))
|
2009-05-26 01:27:19 +04:00
|
|
|
return std::max(ABIAlign, (unsigned)getTypeSize(T));
|
|
|
|
|
2009-01-27 21:08:34 +03:00
|
|
|
return ABIAlign;
|
|
|
|
}
|
|
|
|
|
2010-08-21 01:21:08 +04:00
|
|
|
/// DeepCollectObjCIvars -
|
|
|
|
/// This routine first collects all declared, but not synthesized, ivars in
|
|
|
|
/// super class and then collects all ivars, including those synthesized for
|
|
|
|
/// current class. This routine is used for implementation of current class
|
|
|
|
/// when all ivars, declared and synthesized are known.
|
2009-05-12 22:14:29 +04:00
|
|
|
///
|
2010-08-21 01:21:08 +04:00
|
|
|
void ASTContext::DeepCollectObjCIvars(const ObjCInterfaceDecl *OI,
|
|
|
|
bool leafClass,
|
2011-07-22 06:08:32 +04:00
|
|
|
SmallVectorImpl<const ObjCIvarDecl*> &Ivars) const {
|
2010-08-21 01:21:08 +04:00
|
|
|
if (const ObjCInterfaceDecl *SuperClass = OI->getSuperClass())
|
|
|
|
DeepCollectObjCIvars(SuperClass, false, Ivars);
|
|
|
|
if (!leafClass) {
|
|
|
|
for (ObjCInterfaceDecl::ivar_iterator I = OI->ivar_begin(),
|
|
|
|
E = OI->ivar_end(); I != E; ++I)
|
2010-02-19 03:31:17 +03:00
|
|
|
Ivars.push_back(*I);
|
2011-08-18 03:08:45 +04:00
|
|
|
} else {
|
2011-06-28 22:05:25 +04:00
|
|
|
ObjCInterfaceDecl *IDecl = const_cast<ObjCInterfaceDecl *>(OI);
|
2011-07-22 06:08:32 +04:00
|
|
|
for (const ObjCIvarDecl *Iv = IDecl->all_declared_ivar_begin(); Iv;
|
2011-06-28 22:05:25 +04:00
|
|
|
Iv= Iv->getNextIvar())
|
|
|
|
Ivars.push_back(Iv);
|
|
|
|
}
|
2009-05-12 22:14:29 +04:00
|
|
|
}
|
|
|
|
|
2009-10-30 04:13:23 +03:00
|
|
|
/// CollectInheritedProtocols - Collect all protocols in current class and
|
|
|
|
/// those inherited by it.
|
|
|
|
void ASTContext::CollectInheritedProtocols(const Decl *CDecl,
|
2010-02-12 22:27:33 +03:00
|
|
|
llvm::SmallPtrSet<ObjCProtocolDecl*, 8> &Protocols) {
|
2009-10-30 04:13:23 +03:00
|
|
|
if (const ObjCInterfaceDecl *OI = dyn_cast<ObjCInterfaceDecl>(CDecl)) {
|
2010-09-01 05:21:15 +04:00
|
|
|
// We can use protocol_iterator here instead of
|
|
|
|
// all_referenced_protocol_iterator since we are walking all categories.
|
|
|
|
for (ObjCInterfaceDecl::all_protocol_iterator P = OI->all_referenced_protocol_begin(),
|
|
|
|
PE = OI->all_referenced_protocol_end(); P != PE; ++P) {
|
2009-10-30 04:13:23 +03:00
|
|
|
ObjCProtocolDecl *Proto = (*P);
|
2012-01-01 22:09:12 +04:00
|
|
|
Protocols.insert(Proto->getCanonicalDecl());
|
2009-10-30 04:13:23 +03:00
|
|
|
for (ObjCProtocolDecl::protocol_iterator P = Proto->protocol_begin(),
|
2010-02-25 21:24:33 +03:00
|
|
|
PE = Proto->protocol_end(); P != PE; ++P) {
|
2012-01-01 22:09:12 +04:00
|
|
|
Protocols.insert((*P)->getCanonicalDecl());
|
2009-10-30 04:13:23 +03:00
|
|
|
CollectInheritedProtocols(*P, Protocols);
|
|
|
|
}
|
2010-02-25 21:24:33 +03:00
|
|
|
}
|
2009-10-30 04:13:23 +03:00
|
|
|
|
|
|
|
// Categories of this Interface.
|
|
|
|
for (const ObjCCategoryDecl *CDeclChain = OI->getCategoryList();
|
|
|
|
CDeclChain; CDeclChain = CDeclChain->getNextClassCategory())
|
|
|
|
CollectInheritedProtocols(CDeclChain, Protocols);
|
|
|
|
if (ObjCInterfaceDecl *SD = OI->getSuperClass())
|
|
|
|
while (SD) {
|
|
|
|
CollectInheritedProtocols(SD, Protocols);
|
|
|
|
SD = SD->getSuperClass();
|
|
|
|
}
|
2010-04-27 21:47:25 +04:00
|
|
|
} else if (const ObjCCategoryDecl *OC = dyn_cast<ObjCCategoryDecl>(CDecl)) {
|
2010-09-01 05:21:15 +04:00
|
|
|
for (ObjCCategoryDecl::protocol_iterator P = OC->protocol_begin(),
|
2009-10-30 04:13:23 +03:00
|
|
|
PE = OC->protocol_end(); P != PE; ++P) {
|
|
|
|
ObjCProtocolDecl *Proto = (*P);
|
2012-01-01 22:09:12 +04:00
|
|
|
Protocols.insert(Proto->getCanonicalDecl());
|
2009-10-30 04:13:23 +03:00
|
|
|
for (ObjCProtocolDecl::protocol_iterator P = Proto->protocol_begin(),
|
|
|
|
PE = Proto->protocol_end(); P != PE; ++P)
|
|
|
|
CollectInheritedProtocols(*P, Protocols);
|
|
|
|
}
|
2010-04-27 21:47:25 +04:00
|
|
|
} else if (const ObjCProtocolDecl *OP = dyn_cast<ObjCProtocolDecl>(CDecl)) {
|
2009-10-30 04:13:23 +03:00
|
|
|
for (ObjCProtocolDecl::protocol_iterator P = OP->protocol_begin(),
|
|
|
|
PE = OP->protocol_end(); P != PE; ++P) {
|
|
|
|
ObjCProtocolDecl *Proto = (*P);
|
2012-01-01 22:09:12 +04:00
|
|
|
Protocols.insert(Proto->getCanonicalDecl());
|
2009-10-30 04:13:23 +03:00
|
|
|
for (ObjCProtocolDecl::protocol_iterator P = Proto->protocol_begin(),
|
|
|
|
PE = Proto->protocol_end(); P != PE; ++P)
|
|
|
|
CollectInheritedProtocols(*P, Protocols);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-01-12 12:06:06 +03:00
|
|
|
unsigned ASTContext::CountNonClassIvars(const ObjCInterfaceDecl *OI) const {
|
2010-03-22 21:25:57 +03:00
|
|
|
unsigned count = 0;
|
|
|
|
// Count ivars declared in class extension.
|
2010-06-23 03:20:40 +04:00
|
|
|
for (const ObjCCategoryDecl *CDecl = OI->getFirstClassExtension(); CDecl;
|
|
|
|
CDecl = CDecl->getNextClassExtension())
|
2010-04-27 21:47:25 +04:00
|
|
|
count += CDecl->ivar_size();
|
|
|
|
|
2010-03-22 21:25:57 +03:00
|
|
|
// Count ivar defined in this class's implementation. This
|
|
|
|
// includes synthesized ivars.
|
|
|
|
if (ObjCImplementationDecl *ImplDecl = OI->getImplementation())
|
2010-04-27 21:47:25 +04:00
|
|
|
count += ImplDecl->ivar_size();
|
|
|
|
|
2009-06-04 05:19:09 +04:00
|
|
|
return count;
|
|
|
|
}
|
|
|
|
|
2012-02-03 09:58:16 +04:00
|
|
|
bool ASTContext::isSentinelNullExpr(const Expr *E) {
|
|
|
|
if (!E)
|
|
|
|
return false;
|
|
|
|
|
|
|
|
// nullptr_t is always treated as null.
|
|
|
|
if (E->getType()->isNullPtrType()) return true;
|
|
|
|
|
|
|
|
if (E->getType()->isAnyPointerType() &&
|
|
|
|
E->IgnoreParenCasts()->isNullPointerConstant(*this,
|
|
|
|
Expr::NPC_ValueDependentIsNull))
|
|
|
|
return true;
|
|
|
|
|
|
|
|
// Unfortunately, __null has type 'int'.
|
|
|
|
if (isa<GNUNullExpr>(E)) return true;
|
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2009-07-21 04:05:53 +04:00
|
|
|
/// \brief Get the implementation of ObjCInterfaceDecl,or NULL if none exists.
|
|
|
|
ObjCImplementationDecl *ASTContext::getObjCImplementation(ObjCInterfaceDecl *D) {
|
|
|
|
llvm::DenseMap<ObjCContainerDecl*, ObjCImplDecl*>::iterator
|
|
|
|
I = ObjCImpls.find(D);
|
|
|
|
if (I != ObjCImpls.end())
|
|
|
|
return cast<ObjCImplementationDecl>(I->second);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
/// \brief Get the implementation of ObjCCategoryDecl, or NULL if none exists.
|
|
|
|
ObjCCategoryImplDecl *ASTContext::getObjCImplementation(ObjCCategoryDecl *D) {
|
|
|
|
llvm::DenseMap<ObjCContainerDecl*, ObjCImplDecl*>::iterator
|
|
|
|
I = ObjCImpls.find(D);
|
|
|
|
if (I != ObjCImpls.end())
|
|
|
|
return cast<ObjCCategoryImplDecl>(I->second);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
/// \brief Set the implementation of ObjCInterfaceDecl.
|
|
|
|
void ASTContext::setObjCImplementation(ObjCInterfaceDecl *IFaceD,
|
|
|
|
ObjCImplementationDecl *ImplD) {
|
|
|
|
assert(IFaceD && ImplD && "Passed null params");
|
|
|
|
ObjCImpls[IFaceD] = ImplD;
|
|
|
|
}
|
|
|
|
/// \brief Set the implementation of ObjCCategoryDecl.
|
|
|
|
void ASTContext::setObjCImplementation(ObjCCategoryDecl *CatD,
|
|
|
|
ObjCCategoryImplDecl *ImplD) {
|
|
|
|
assert(CatD && ImplD && "Passed null params");
|
|
|
|
ObjCImpls[CatD] = ImplD;
|
|
|
|
}
|
|
|
|
|
2011-11-01 21:14:12 +04:00
|
|
|
ObjCInterfaceDecl *ASTContext::getObjContainingInterface(NamedDecl *ND) const {
|
|
|
|
if (ObjCInterfaceDecl *ID = dyn_cast<ObjCInterfaceDecl>(ND->getDeclContext()))
|
|
|
|
return ID;
|
|
|
|
if (ObjCCategoryDecl *CD = dyn_cast<ObjCCategoryDecl>(ND->getDeclContext()))
|
|
|
|
return CD->getClassInterface();
|
|
|
|
if (ObjCImplDecl *IMD = dyn_cast<ObjCImplDecl>(ND->getDeclContext()))
|
|
|
|
return IMD->getClassInterface();
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2010-12-02 01:29:46 +03:00
|
|
|
/// \brief Get the copy initialization expression of VarDecl,or NULL if
|
|
|
|
/// none exists.
|
2010-12-02 20:02:11 +03:00
|
|
|
Expr *ASTContext::getBlockVarCopyInits(const VarDecl*VD) {
|
2010-12-06 20:28:17 +03:00
|
|
|
assert(VD && "Passed null params");
|
|
|
|
assert(VD->hasAttr<BlocksAttr>() &&
|
|
|
|
"getBlockVarCopyInits - not __block var");
|
2010-12-02 20:02:11 +03:00
|
|
|
llvm::DenseMap<const VarDecl*, Expr*>::iterator
|
2010-12-06 20:28:17 +03:00
|
|
|
I = BlockVarCopyInits.find(VD);
|
2010-12-02 01:29:46 +03:00
|
|
|
return (I != BlockVarCopyInits.end()) ? cast<Expr>(I->second) : 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
/// \brief Set the copy inialization expression of a block var decl.
|
|
|
|
void ASTContext::setBlockVarCopyInits(VarDecl*VD, Expr* Init) {
|
|
|
|
assert(VD && Init && "Passed null params");
|
2010-12-06 20:28:17 +03:00
|
|
|
assert(VD->hasAttr<BlocksAttr>() &&
|
|
|
|
"setBlockVarCopyInits - not __block var");
|
2010-12-02 01:29:46 +03:00
|
|
|
BlockVarCopyInits[VD] = Init;
|
|
|
|
}
|
|
|
|
|
2009-12-07 05:54:59 +03:00
|
|
|
/// \brief Allocate an uninitialized TypeSourceInfo.
|
2009-08-19 05:27:32 +04:00
|
|
|
///
|
2009-12-07 05:54:59 +03:00
|
|
|
/// The caller should initialize the memory held by TypeSourceInfo using
|
2009-08-19 05:27:32 +04:00
|
|
|
/// the TypeLoc wrappers.
|
|
|
|
///
|
|
|
|
/// \param T the type that will be the basis for type source info. This type
|
|
|
|
/// should refer to how the declarator was written in source code, not to
|
|
|
|
/// what type semantic analysis resolved the declarator to.
|
2009-12-07 05:54:59 +03:00
|
|
|
TypeSourceInfo *ASTContext::CreateTypeSourceInfo(QualType T,
|
2011-01-12 12:06:06 +03:00
|
|
|
unsigned DataSize) const {
|
2009-10-21 04:23:54 +04:00
|
|
|
if (!DataSize)
|
|
|
|
DataSize = TypeLoc::getFullDataSizeForType(T);
|
|
|
|
else
|
|
|
|
assert(DataSize == TypeLoc::getFullDataSizeForType(T) &&
|
2009-12-07 05:54:59 +03:00
|
|
|
"incorrect data size provided to CreateTypeSourceInfo!");
|
2009-10-21 04:23:54 +04:00
|
|
|
|
2009-12-07 05:54:59 +03:00
|
|
|
TypeSourceInfo *TInfo =
|
|
|
|
(TypeSourceInfo*)BumpAlloc.Allocate(sizeof(TypeSourceInfo) + DataSize, 8);
|
|
|
|
new (TInfo) TypeSourceInfo(T);
|
|
|
|
return TInfo;
|
2009-08-19 05:27:32 +04:00
|
|
|
}
|
|
|
|
|
2009-12-07 05:54:59 +03:00
|
|
|
TypeSourceInfo *ASTContext::getTrivialTypeSourceInfo(QualType T,
|
2011-01-19 23:10:05 +03:00
|
|
|
SourceLocation L) const {
|
2009-12-07 05:54:59 +03:00
|
|
|
TypeSourceInfo *DI = CreateTypeSourceInfo(T);
|
2011-01-25 22:13:18 +03:00
|
|
|
DI->getTypeLoc().initialize(const_cast<ASTContext &>(*this), L);
|
2009-10-24 01:14:09 +04:00
|
|
|
return DI;
|
|
|
|
}
|
|
|
|
|
2009-05-03 14:38:35 +04:00
|
|
|
const ASTRecordLayout &
|
2011-01-12 12:06:06 +03:00
|
|
|
ASTContext::getASTObjCInterfaceLayout(const ObjCInterfaceDecl *D) const {
|
2009-05-03 14:38:35 +04:00
|
|
|
return getObjCLayout(D, 0);
|
|
|
|
}
|
|
|
|
|
|
|
|
const ASTRecordLayout &
|
2011-01-12 12:06:06 +03:00
|
|
|
ASTContext::getASTObjCImplementationLayout(
|
|
|
|
const ObjCImplementationDecl *D) const {
|
2009-05-03 14:38:35 +04:00
|
|
|
return getObjCLayout(D->getClassInterface(), D);
|
|
|
|
}
|
|
|
|
|
2007-07-14 02:13:22 +04:00
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
// Type creation/memoization methods
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
|
2011-01-12 12:06:06 +03:00
|
|
|
QualType
|
2011-01-19 13:06:00 +03:00
|
|
|
ASTContext::getExtQualType(const Type *baseType, Qualifiers quals) const {
|
|
|
|
unsigned fastQuals = quals.getFastQualifiers();
|
|
|
|
quals.removeFastQualifiers();
|
2009-09-24 23:53:00 +04:00
|
|
|
|
|
|
|
// Check if we've already instantiated this type.
|
|
|
|
llvm::FoldingSetNodeID ID;
|
2011-01-19 13:06:00 +03:00
|
|
|
ExtQuals::Profile(ID, baseType, quals);
|
|
|
|
void *insertPos = 0;
|
|
|
|
if (ExtQuals *eq = ExtQualNodes.FindNodeOrInsertPos(ID, insertPos)) {
|
|
|
|
assert(eq->getQualifiers() == quals);
|
|
|
|
return QualType(eq, fastQuals);
|
2009-09-24 23:53:00 +04:00
|
|
|
}
|
|
|
|
|
2011-01-19 13:06:00 +03:00
|
|
|
// If the base type is not canonical, make the appropriate canonical type.
|
|
|
|
QualType canon;
|
|
|
|
if (!baseType->isCanonicalUnqualified()) {
|
|
|
|
SplitQualType canonSplit = baseType->getCanonicalTypeInternal().split();
|
2012-02-08 04:46:36 +04:00
|
|
|
canonSplit.Quals.addConsistentQualifiers(quals);
|
|
|
|
canon = getExtQualType(canonSplit.Ty, canonSplit.Quals);
|
2011-01-19 13:06:00 +03:00
|
|
|
|
|
|
|
// Re-find the insert position.
|
|
|
|
(void) ExtQualNodes.FindNodeOrInsertPos(ID, insertPos);
|
|
|
|
}
|
|
|
|
|
|
|
|
ExtQuals *eq = new (*this, TypeAlignment) ExtQuals(baseType, canon, quals);
|
|
|
|
ExtQualNodes.InsertNode(eq, insertPos);
|
|
|
|
return QualType(eq, fastQuals);
|
2009-09-24 23:53:00 +04:00
|
|
|
}
|
|
|
|
|
2011-01-12 12:06:06 +03:00
|
|
|
QualType
|
|
|
|
ASTContext::getAddrSpaceQualType(QualType T, unsigned AddressSpace) const {
|
2008-04-07 02:59:24 +04:00
|
|
|
QualType CanT = getCanonicalType(T);
|
|
|
|
if (CanT.getAddressSpace() == AddressSpace)
|
2008-02-20 23:55:12 +03:00
|
|
|
return T;
|
2009-02-19 01:53:11 +03:00
|
|
|
|
2009-09-24 23:53:00 +04:00
|
|
|
// If we are composing extended qualifiers together, merge together
|
|
|
|
// into one ExtQuals node.
|
|
|
|
QualifierCollector Quals;
|
|
|
|
const Type *TypeNode = Quals.strip(T);
|
2009-09-09 19:08:12 +04:00
|
|
|
|
2009-09-24 23:53:00 +04:00
|
|
|
// If this type already has an address space specified, it cannot get
|
|
|
|
// another one.
|
|
|
|
assert(!Quals.hasAddressSpace() &&
|
|
|
|
"Type cannot be in multiple addr spaces!");
|
|
|
|
Quals.addAddressSpace(AddressSpace);
|
2009-09-09 19:08:12 +04:00
|
|
|
|
2009-09-24 23:53:00 +04:00
|
|
|
return getExtQualType(TypeNode, Quals);
|
2008-02-04 05:31:56 +03:00
|
|
|
}
|
|
|
|
|
2009-02-19 01:53:11 +03:00
|
|
|
QualType ASTContext::getObjCGCQualType(QualType T,
|
2011-01-12 12:06:06 +03:00
|
|
|
Qualifiers::GC GCAttr) const {
|
2009-02-18 08:09:49 +03:00
|
|
|
QualType CanT = getCanonicalType(T);
|
2009-02-19 01:53:11 +03:00
|
|
|
if (CanT.getObjCGCAttr() == GCAttr)
|
2009-02-18 08:09:49 +03:00
|
|
|
return T;
|
2009-09-09 19:08:12 +04:00
|
|
|
|
2010-12-24 05:08:15 +03:00
|
|
|
if (const PointerType *ptr = T->getAs<PointerType>()) {
|
|
|
|
QualType Pointee = ptr->getPointeeType();
|
2009-07-14 22:25:06 +04:00
|
|
|
if (Pointee->isAnyPointerType()) {
|
2009-06-03 21:15:17 +04:00
|
|
|
QualType ResultType = getObjCGCQualType(Pointee, GCAttr);
|
|
|
|
return getPointerType(ResultType);
|
|
|
|
}
|
|
|
|
}
|
2009-09-09 19:08:12 +04:00
|
|
|
|
2009-09-24 23:53:00 +04:00
|
|
|
// If we are composing extended qualifiers together, merge together
|
|
|
|
// into one ExtQuals node.
|
|
|
|
QualifierCollector Quals;
|
|
|
|
const Type *TypeNode = Quals.strip(T);
|
2009-09-09 19:08:12 +04:00
|
|
|
|
2009-09-24 23:53:00 +04:00
|
|
|
// If this type already has an ObjCGC specified, it cannot get
|
|
|
|
// another one.
|
|
|
|
assert(!Quals.hasObjCGCAttr() &&
|
|
|
|
"Type cannot have multiple ObjCGCs!");
|
|
|
|
Quals.addObjCGCAttr(GCAttr);
|
2009-09-09 19:08:12 +04:00
|
|
|
|
2009-09-24 23:53:00 +04:00
|
|
|
return getExtQualType(TypeNode, Quals);
|
2009-02-18 08:09:49 +03:00
|
|
|
}
|
2007-07-14 02:13:22 +04:00
|
|
|
|
2010-12-19 05:44:49 +03:00
|
|
|
const FunctionType *ASTContext::adjustFunctionType(const FunctionType *T,
|
|
|
|
FunctionType::ExtInfo Info) {
|
|
|
|
if (T->getExtInfo() == Info)
|
|
|
|
return T;
|
|
|
|
|
|
|
|
QualType Result;
|
|
|
|
if (const FunctionNoProtoType *FNPT = dyn_cast<FunctionNoProtoType>(T)) {
|
|
|
|
Result = getFunctionNoProtoType(FNPT->getResultType(), Info);
|
|
|
|
} else {
|
|
|
|
const FunctionProtoType *FPT = cast<FunctionProtoType>(T);
|
|
|
|
FunctionProtoType::ExtProtoInfo EPI = FPT->getExtProtoInfo();
|
|
|
|
EPI.ExtInfo = Info;
|
|
|
|
Result = getFunctionType(FPT->getResultType(), FPT->arg_type_begin(),
|
|
|
|
FPT->getNumArgs(), EPI);
|
|
|
|
}
|
|
|
|
|
|
|
|
return cast<FunctionType>(Result.getTypePtr());
|
|
|
|
}
|
|
|
|
|
2007-07-11 21:01:13 +04:00
|
|
|
/// getComplexType - Return the uniqued reference to the type for a complex
|
|
|
|
/// number with the specified element type.
|
2011-01-12 12:06:06 +03:00
|
|
|
QualType ASTContext::getComplexType(QualType T) const {
|
2007-07-11 21:01:13 +04:00
|
|
|
// Unique pointers, to guarantee there is only one pointer of a particular
|
|
|
|
// structure.
|
|
|
|
llvm::FoldingSetNodeID ID;
|
|
|
|
ComplexType::Profile(ID, T);
|
2009-09-09 19:08:12 +04:00
|
|
|
|
2007-07-11 21:01:13 +04:00
|
|
|
void *InsertPos = 0;
|
|
|
|
if (ComplexType *CT = ComplexTypes.FindNodeOrInsertPos(ID, InsertPos))
|
|
|
|
return QualType(CT, 0);
|
2009-09-09 19:08:12 +04:00
|
|
|
|
2007-07-11 21:01:13 +04:00
|
|
|
// If the pointee type isn't canonical, this won't be a canonical type either,
|
|
|
|
// so fill in the canonical type field.
|
|
|
|
QualType Canonical;
|
2009-10-23 00:10:53 +04:00
|
|
|
if (!T.isCanonical()) {
|
2008-04-07 02:59:24 +04:00
|
|
|
Canonical = getComplexType(getCanonicalType(T));
|
2009-09-09 19:08:12 +04:00
|
|
|
|
2007-07-11 21:01:13 +04:00
|
|
|
// Get the new insert position for the node we care about.
|
|
|
|
ComplexType *NewIP = ComplexTypes.FindNodeOrInsertPos(ID, InsertPos);
|
2010-12-23 04:01:28 +03:00
|
|
|
assert(NewIP == 0 && "Shouldn't be in the map!"); (void)NewIP;
|
2007-07-11 21:01:13 +04:00
|
|
|
}
|
2009-09-25 03:30:46 +04:00
|
|
|
ComplexType *New = new (*this, TypeAlignment) ComplexType(T, Canonical);
|
2007-07-11 21:01:13 +04:00
|
|
|
Types.push_back(New);
|
|
|
|
ComplexTypes.InsertNode(New, InsertPos);
|
|
|
|
return QualType(New, 0);
|
|
|
|
}
|
|
|
|
|
|
|
|
/// getPointerType - Return the uniqued reference to the type for a pointer to
|
|
|
|
/// the specified type.
|
2011-01-12 12:06:06 +03:00
|
|
|
QualType ASTContext::getPointerType(QualType T) const {
|
2007-07-11 21:01:13 +04:00
|
|
|
// Unique pointers, to guarantee there is only one pointer of a particular
|
|
|
|
// structure.
|
|
|
|
llvm::FoldingSetNodeID ID;
|
|
|
|
PointerType::Profile(ID, T);
|
2009-09-09 19:08:12 +04:00
|
|
|
|
2007-07-11 21:01:13 +04:00
|
|
|
void *InsertPos = 0;
|
|
|
|
if (PointerType *PT = PointerTypes.FindNodeOrInsertPos(ID, InsertPos))
|
|
|
|
return QualType(PT, 0);
|
2009-09-09 19:08:12 +04:00
|
|
|
|
2007-07-11 21:01:13 +04:00
|
|
|
// If the pointee type isn't canonical, this won't be a canonical type either,
|
|
|
|
// so fill in the canonical type field.
|
|
|
|
QualType Canonical;
|
2009-10-23 00:10:53 +04:00
|
|
|
if (!T.isCanonical()) {
|
2008-04-07 02:59:24 +04:00
|
|
|
Canonical = getPointerType(getCanonicalType(T));
|
2009-09-09 19:08:12 +04:00
|
|
|
|
2007-07-11 21:01:13 +04:00
|
|
|
// Get the new insert position for the node we care about.
|
|
|
|
PointerType *NewIP = PointerTypes.FindNodeOrInsertPos(ID, InsertPos);
|
2010-12-23 04:01:28 +03:00
|
|
|
assert(NewIP == 0 && "Shouldn't be in the map!"); (void)NewIP;
|
2007-07-11 21:01:13 +04:00
|
|
|
}
|
2009-09-25 03:30:46 +04:00
|
|
|
PointerType *New = new (*this, TypeAlignment) PointerType(T, Canonical);
|
2007-07-11 21:01:13 +04:00
|
|
|
Types.push_back(New);
|
|
|
|
PointerTypes.InsertNode(New, InsertPos);
|
|
|
|
return QualType(New, 0);
|
|
|
|
}
|
|
|
|
|
2009-09-09 19:08:12 +04:00
|
|
|
/// getBlockPointerType - Return the uniqued reference to the type for
|
2008-08-27 20:04:49 +04:00
|
|
|
/// a pointer to the specified block.
|
2011-01-12 12:06:06 +03:00
|
|
|
QualType ASTContext::getBlockPointerType(QualType T) const {
|
2008-08-28 23:20:44 +04:00
|
|
|
assert(T->isFunctionType() && "block of function types only");
|
|
|
|
// Unique pointers, to guarantee there is only one block of a particular
|
2008-08-27 20:04:49 +04:00
|
|
|
// structure.
|
|
|
|
llvm::FoldingSetNodeID ID;
|
|
|
|
BlockPointerType::Profile(ID, T);
|
2009-09-09 19:08:12 +04:00
|
|
|
|
2008-08-27 20:04:49 +04:00
|
|
|
void *InsertPos = 0;
|
|
|
|
if (BlockPointerType *PT =
|
|
|
|
BlockPointerTypes.FindNodeOrInsertPos(ID, InsertPos))
|
|
|
|
return QualType(PT, 0);
|
2009-09-09 19:08:12 +04:00
|
|
|
|
|
|
|
// If the block pointee type isn't canonical, this won't be a canonical
|
2008-08-27 20:04:49 +04:00
|
|
|
// type either so fill in the canonical type field.
|
|
|
|
QualType Canonical;
|
2009-10-23 00:10:53 +04:00
|
|
|
if (!T.isCanonical()) {
|
2008-08-27 20:04:49 +04:00
|
|
|
Canonical = getBlockPointerType(getCanonicalType(T));
|
2009-09-09 19:08:12 +04:00
|
|
|
|
2008-08-27 20:04:49 +04:00
|
|
|
// Get the new insert position for the node we care about.
|
|
|
|
BlockPointerType *NewIP =
|
|
|
|
BlockPointerTypes.FindNodeOrInsertPos(ID, InsertPos);
|
2010-12-23 04:01:28 +03:00
|
|
|
assert(NewIP == 0 && "Shouldn't be in the map!"); (void)NewIP;
|
2008-08-27 20:04:49 +04:00
|
|
|
}
|
2009-09-25 03:30:46 +04:00
|
|
|
BlockPointerType *New
|
|
|
|
= new (*this, TypeAlignment) BlockPointerType(T, Canonical);
|
2008-08-27 20:04:49 +04:00
|
|
|
Types.push_back(New);
|
|
|
|
BlockPointerTypes.InsertNode(New, InsertPos);
|
|
|
|
return QualType(New, 0);
|
|
|
|
}
|
|
|
|
|
2009-03-17 02:22:08 +03:00
|
|
|
/// getLValueReferenceType - Return the uniqued reference to the type for an
|
|
|
|
/// lvalue reference to the specified type.
|
2011-01-12 12:06:06 +03:00
|
|
|
QualType
|
|
|
|
ASTContext::getLValueReferenceType(QualType T, bool SpelledAsLValue) const {
|
2011-05-22 02:16:50 +04:00
|
|
|
assert(getCanonicalType(T) != OverloadTy &&
|
|
|
|
"Unresolved overloaded function type");
|
|
|
|
|
2007-07-11 21:01:13 +04:00
|
|
|
// Unique pointers, to guarantee there is only one pointer of a particular
|
|
|
|
// structure.
|
|
|
|
llvm::FoldingSetNodeID ID;
|
2009-10-23 02:37:11 +04:00
|
|
|
ReferenceType::Profile(ID, T, SpelledAsLValue);
|
2007-07-11 21:01:13 +04:00
|
|
|
|
|
|
|
void *InsertPos = 0;
|
2009-03-17 02:22:08 +03:00
|
|
|
if (LValueReferenceType *RT =
|
|
|
|
LValueReferenceTypes.FindNodeOrInsertPos(ID, InsertPos))
|
2007-07-11 21:01:13 +04:00
|
|
|
return QualType(RT, 0);
|
2009-03-17 02:22:08 +03:00
|
|
|
|
2009-10-23 02:37:11 +04:00
|
|
|
const ReferenceType *InnerRef = T->getAs<ReferenceType>();
|
|
|
|
|
2007-07-11 21:01:13 +04:00
|
|
|
// If the referencee type isn't canonical, this won't be a canonical type
|
|
|
|
// either, so fill in the canonical type field.
|
|
|
|
QualType Canonical;
|
2009-10-23 02:37:11 +04:00
|
|
|
if (!SpelledAsLValue || InnerRef || !T.isCanonical()) {
|
|
|
|
QualType PointeeType = (InnerRef ? InnerRef->getPointeeType() : T);
|
|
|
|
Canonical = getLValueReferenceType(getCanonicalType(PointeeType));
|
2009-03-17 02:22:08 +03:00
|
|
|
|
|
|
|
// Get the new insert position for the node we care about.
|
|
|
|
LValueReferenceType *NewIP =
|
|
|
|
LValueReferenceTypes.FindNodeOrInsertPos(ID, InsertPos);
|
2010-12-23 04:01:28 +03:00
|
|
|
assert(NewIP == 0 && "Shouldn't be in the map!"); (void)NewIP;
|
2009-03-17 02:22:08 +03:00
|
|
|
}
|
|
|
|
|
2009-09-25 03:30:46 +04:00
|
|
|
LValueReferenceType *New
|
2009-10-23 02:37:11 +04:00
|
|
|
= new (*this, TypeAlignment) LValueReferenceType(T, Canonical,
|
|
|
|
SpelledAsLValue);
|
2009-03-17 02:22:08 +03:00
|
|
|
Types.push_back(New);
|
|
|
|
LValueReferenceTypes.InsertNode(New, InsertPos);
|
2009-10-23 02:37:11 +04:00
|
|
|
|
2009-03-17 02:22:08 +03:00
|
|
|
return QualType(New, 0);
|
|
|
|
}
|
|
|
|
|
|
|
|
/// getRValueReferenceType - Return the uniqued reference to the type for an
|
|
|
|
/// rvalue reference to the specified type.
|
2011-01-12 12:06:06 +03:00
|
|
|
QualType ASTContext::getRValueReferenceType(QualType T) const {
|
2009-03-17 02:22:08 +03:00
|
|
|
// Unique pointers, to guarantee there is only one pointer of a particular
|
|
|
|
// structure.
|
|
|
|
llvm::FoldingSetNodeID ID;
|
2009-10-23 02:37:11 +04:00
|
|
|
ReferenceType::Profile(ID, T, false);
|
2009-03-17 02:22:08 +03:00
|
|
|
|
|
|
|
void *InsertPos = 0;
|
|
|
|
if (RValueReferenceType *RT =
|
|
|
|
RValueReferenceTypes.FindNodeOrInsertPos(ID, InsertPos))
|
|
|
|
return QualType(RT, 0);
|
|
|
|
|
2009-10-23 02:37:11 +04:00
|
|
|
const ReferenceType *InnerRef = T->getAs<ReferenceType>();
|
|
|
|
|
2009-03-17 02:22:08 +03:00
|
|
|
// If the referencee type isn't canonical, this won't be a canonical type
|
|
|
|
// either, so fill in the canonical type field.
|
|
|
|
QualType Canonical;
|
2009-10-23 02:37:11 +04:00
|
|
|
if (InnerRef || !T.isCanonical()) {
|
|
|
|
QualType PointeeType = (InnerRef ? InnerRef->getPointeeType() : T);
|
|
|
|
Canonical = getRValueReferenceType(getCanonicalType(PointeeType));
|
2009-03-17 02:22:08 +03:00
|
|
|
|
2007-07-11 21:01:13 +04:00
|
|
|
// Get the new insert position for the node we care about.
|
2009-03-17 02:22:08 +03:00
|
|
|
RValueReferenceType *NewIP =
|
|
|
|
RValueReferenceTypes.FindNodeOrInsertPos(ID, InsertPos);
|
2010-12-23 04:01:28 +03:00
|
|
|
assert(NewIP == 0 && "Shouldn't be in the map!"); (void)NewIP;
|
2007-07-11 21:01:13 +04:00
|
|
|
}
|
|
|
|
|
2009-09-25 03:30:46 +04:00
|
|
|
RValueReferenceType *New
|
|
|
|
= new (*this, TypeAlignment) RValueReferenceType(T, Canonical);
|
2007-07-11 21:01:13 +04:00
|
|
|
Types.push_back(New);
|
2009-03-17 02:22:08 +03:00
|
|
|
RValueReferenceTypes.InsertNode(New, InsertPos);
|
2007-07-11 21:01:13 +04:00
|
|
|
return QualType(New, 0);
|
|
|
|
}
|
|
|
|
|
2009-01-25 00:16:55 +03:00
|
|
|
/// getMemberPointerType - Return the uniqued reference to the type for a
|
|
|
|
/// member pointer to the specified type, in the specified class.
|
2011-01-12 12:06:06 +03:00
|
|
|
QualType ASTContext::getMemberPointerType(QualType T, const Type *Cls) const {
|
2009-01-25 00:16:55 +03:00
|
|
|
// Unique pointers, to guarantee there is only one pointer of a particular
|
|
|
|
// structure.
|
|
|
|
llvm::FoldingSetNodeID ID;
|
|
|
|
MemberPointerType::Profile(ID, T, Cls);
|
|
|
|
|
|
|
|
void *InsertPos = 0;
|
|
|
|
if (MemberPointerType *PT =
|
|
|
|
MemberPointerTypes.FindNodeOrInsertPos(ID, InsertPos))
|
|
|
|
return QualType(PT, 0);
|
|
|
|
|
|
|
|
// If the pointee or class type isn't canonical, this won't be a canonical
|
|
|
|
// type either, so fill in the canonical type field.
|
|
|
|
QualType Canonical;
|
2009-11-04 19:49:01 +03:00
|
|
|
if (!T.isCanonical() || !Cls->isCanonicalUnqualified()) {
|
2009-01-25 00:16:55 +03:00
|
|
|
Canonical = getMemberPointerType(getCanonicalType(T),getCanonicalType(Cls));
|
|
|
|
|
|
|
|
// Get the new insert position for the node we care about.
|
|
|
|
MemberPointerType *NewIP =
|
|
|
|
MemberPointerTypes.FindNodeOrInsertPos(ID, InsertPos);
|
2010-12-23 04:01:28 +03:00
|
|
|
assert(NewIP == 0 && "Shouldn't be in the map!"); (void)NewIP;
|
2009-01-25 00:16:55 +03:00
|
|
|
}
|
2009-09-25 03:30:46 +04:00
|
|
|
MemberPointerType *New
|
|
|
|
= new (*this, TypeAlignment) MemberPointerType(T, Cls, Canonical);
|
2009-01-25 00:16:55 +03:00
|
|
|
Types.push_back(New);
|
|
|
|
MemberPointerTypes.InsertNode(New, InsertPos);
|
|
|
|
return QualType(New, 0);
|
|
|
|
}
|
|
|
|
|
2009-09-09 19:08:12 +04:00
|
|
|
/// getConstantArrayType - Return the unique reference to the type for an
|
2007-08-30 05:06:46 +04:00
|
|
|
/// array of the specified element type.
|
2009-09-09 19:08:12 +04:00
|
|
|
QualType ASTContext::getConstantArrayType(QualType EltTy,
|
2009-05-13 08:12:56 +04:00
|
|
|
const llvm::APInt &ArySizeIn,
|
2007-08-30 22:10:14 +04:00
|
|
|
ArrayType::ArraySizeModifier ASM,
|
2011-01-27 22:55:10 +03:00
|
|
|
unsigned IndexTypeQuals) const {
|
2009-11-05 18:52:31 +03:00
|
|
|
assert((EltTy->isDependentType() ||
|
|
|
|
EltTy->isIncompleteType() || EltTy->isConstantSizeType()) &&
|
2009-05-30 00:17:55 +04:00
|
|
|
"Constant array of VLAs is illegal!");
|
|
|
|
|
2009-05-13 08:12:56 +04:00
|
|
|
// Convert the array size into a canonical width matching the pointer size for
|
|
|
|
// the target.
|
|
|
|
llvm::APInt ArySize(ArySizeIn);
|
2010-12-07 11:25:34 +03:00
|
|
|
ArySize =
|
2011-09-02 04:18:52 +04:00
|
|
|
ArySize.zextOrTrunc(Target->getPointerWidth(getTargetAddressSpace(EltTy)));
|
2009-09-09 19:08:12 +04:00
|
|
|
|
2007-07-11 21:01:13 +04:00
|
|
|
llvm::FoldingSetNodeID ID;
|
2011-01-27 22:55:10 +03:00
|
|
|
ConstantArrayType::Profile(ID, EltTy, ArySize, ASM, IndexTypeQuals);
|
2009-09-09 19:08:12 +04:00
|
|
|
|
2007-07-11 21:01:13 +04:00
|
|
|
void *InsertPos = 0;
|
2009-09-09 19:08:12 +04:00
|
|
|
if (ConstantArrayType *ATP =
|
2007-10-31 20:10:13 +03:00
|
|
|
ConstantArrayTypes.FindNodeOrInsertPos(ID, InsertPos))
|
2007-07-11 21:01:13 +04:00
|
|
|
return QualType(ATP, 0);
|
2009-09-09 19:08:12 +04:00
|
|
|
|
2011-01-19 13:06:00 +03:00
|
|
|
// If the element type isn't canonical or has qualifiers, this won't
|
|
|
|
// be a canonical type either, so fill in the canonical type field.
|
|
|
|
QualType Canon;
|
|
|
|
if (!EltTy.isCanonical() || EltTy.hasLocalQualifiers()) {
|
|
|
|
SplitQualType canonSplit = getCanonicalType(EltTy).split();
|
2012-02-08 04:46:36 +04:00
|
|
|
Canon = getConstantArrayType(QualType(canonSplit.Ty, 0), ArySize,
|
2011-01-27 22:55:10 +03:00
|
|
|
ASM, IndexTypeQuals);
|
2012-02-08 04:46:36 +04:00
|
|
|
Canon = getQualifiedType(Canon, canonSplit.Quals);
|
2011-01-19 13:06:00 +03:00
|
|
|
|
2007-07-11 21:01:13 +04:00
|
|
|
// Get the new insert position for the node we care about.
|
2009-09-09 19:08:12 +04:00
|
|
|
ConstantArrayType *NewIP =
|
2007-10-31 20:10:13 +03:00
|
|
|
ConstantArrayTypes.FindNodeOrInsertPos(ID, InsertPos);
|
2010-12-23 04:01:28 +03:00
|
|
|
assert(NewIP == 0 && "Shouldn't be in the map!"); (void)NewIP;
|
2007-07-11 21:01:13 +04:00
|
|
|
}
|
2009-09-09 19:08:12 +04:00
|
|
|
|
2009-09-25 03:30:46 +04:00
|
|
|
ConstantArrayType *New = new(*this,TypeAlignment)
|
2011-01-27 22:55:10 +03:00
|
|
|
ConstantArrayType(EltTy, Canon, ArySize, ASM, IndexTypeQuals);
|
2007-10-31 20:10:13 +03:00
|
|
|
ConstantArrayTypes.InsertNode(New, InsertPos);
|
2007-07-11 21:01:13 +04:00
|
|
|
Types.push_back(New);
|
|
|
|
return QualType(New, 0);
|
|
|
|
}
|
|
|
|
|
2011-01-18 11:40:38 +03:00
|
|
|
/// getVariableArrayDecayedType - Turns the given type, which may be
|
|
|
|
/// variably-modified, into the corresponding type with all the known
|
|
|
|
/// sizes replaced with [*].
|
|
|
|
QualType ASTContext::getVariableArrayDecayedType(QualType type) const {
|
|
|
|
// Vastly most common case.
|
|
|
|
if (!type->isVariablyModifiedType()) return type;
|
|
|
|
|
|
|
|
QualType result;
|
|
|
|
|
|
|
|
SplitQualType split = type.getSplitDesugaredType();
|
2012-02-08 04:46:36 +04:00
|
|
|
const Type *ty = split.Ty;
|
2011-01-18 11:40:38 +03:00
|
|
|
switch (ty->getTypeClass()) {
|
|
|
|
#define TYPE(Class, Base)
|
|
|
|
#define ABSTRACT_TYPE(Class, Base)
|
|
|
|
#define NON_CANONICAL_TYPE(Class, Base) case Type::Class:
|
|
|
|
#include "clang/AST/TypeNodes.def"
|
|
|
|
llvm_unreachable("didn't desugar past all non-canonical types?");
|
|
|
|
|
|
|
|
// These types should never be variably-modified.
|
|
|
|
case Type::Builtin:
|
|
|
|
case Type::Complex:
|
|
|
|
case Type::Vector:
|
|
|
|
case Type::ExtVector:
|
|
|
|
case Type::DependentSizedExtVector:
|
|
|
|
case Type::ObjCObject:
|
|
|
|
case Type::ObjCInterface:
|
|
|
|
case Type::ObjCObjectPointer:
|
|
|
|
case Type::Record:
|
|
|
|
case Type::Enum:
|
|
|
|
case Type::UnresolvedUsing:
|
|
|
|
case Type::TypeOfExpr:
|
|
|
|
case Type::TypeOf:
|
|
|
|
case Type::Decltype:
|
2011-05-25 02:41:36 +04:00
|
|
|
case Type::UnaryTransform:
|
2011-01-18 11:40:38 +03:00
|
|
|
case Type::DependentName:
|
|
|
|
case Type::InjectedClassName:
|
|
|
|
case Type::TemplateSpecialization:
|
|
|
|
case Type::DependentTemplateSpecialization:
|
|
|
|
case Type::TemplateTypeParm:
|
|
|
|
case Type::SubstTemplateTypeParmPack:
|
2011-02-20 06:19:35 +03:00
|
|
|
case Type::Auto:
|
2011-01-18 11:40:38 +03:00
|
|
|
case Type::PackExpansion:
|
|
|
|
llvm_unreachable("type should never be variably-modified");
|
|
|
|
|
|
|
|
// These types can be variably-modified but should never need to
|
|
|
|
// further decay.
|
|
|
|
case Type::FunctionNoProto:
|
|
|
|
case Type::FunctionProto:
|
|
|
|
case Type::BlockPointer:
|
|
|
|
case Type::MemberPointer:
|
|
|
|
return type;
|
|
|
|
|
|
|
|
// These types can be variably-modified. All these modifications
|
|
|
|
// preserve structure except as noted by comments.
|
|
|
|
// TODO: if we ever care about optimizing VLAs, there are no-op
|
|
|
|
// optimizations available here.
|
|
|
|
case Type::Pointer:
|
|
|
|
result = getPointerType(getVariableArrayDecayedType(
|
|
|
|
cast<PointerType>(ty)->getPointeeType()));
|
|
|
|
break;
|
|
|
|
|
|
|
|
case Type::LValueReference: {
|
|
|
|
const LValueReferenceType *lv = cast<LValueReferenceType>(ty);
|
|
|
|
result = getLValueReferenceType(
|
|
|
|
getVariableArrayDecayedType(lv->getPointeeType()),
|
|
|
|
lv->isSpelledAsLValue());
|
|
|
|
break;
|
2010-09-24 21:30:16 +04:00
|
|
|
}
|
2011-01-18 11:40:38 +03:00
|
|
|
|
|
|
|
case Type::RValueReference: {
|
|
|
|
const RValueReferenceType *lv = cast<RValueReferenceType>(ty);
|
|
|
|
result = getRValueReferenceType(
|
|
|
|
getVariableArrayDecayedType(lv->getPointeeType()));
|
|
|
|
break;
|
2010-09-24 21:30:16 +04:00
|
|
|
}
|
|
|
|
|
2011-10-07 03:00:33 +04:00
|
|
|
case Type::Atomic: {
|
|
|
|
const AtomicType *at = cast<AtomicType>(ty);
|
|
|
|
result = getAtomicType(getVariableArrayDecayedType(at->getValueType()));
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2011-01-18 11:40:38 +03:00
|
|
|
case Type::ConstantArray: {
|
|
|
|
const ConstantArrayType *cat = cast<ConstantArrayType>(ty);
|
|
|
|
result = getConstantArrayType(
|
|
|
|
getVariableArrayDecayedType(cat->getElementType()),
|
|
|
|
cat->getSize(),
|
|
|
|
cat->getSizeModifier(),
|
|
|
|
cat->getIndexTypeCVRQualifiers());
|
|
|
|
break;
|
2010-09-24 21:30:16 +04:00
|
|
|
}
|
|
|
|
|
2011-01-18 11:40:38 +03:00
|
|
|
case Type::DependentSizedArray: {
|
|
|
|
const DependentSizedArrayType *dat = cast<DependentSizedArrayType>(ty);
|
|
|
|
result = getDependentSizedArrayType(
|
|
|
|
getVariableArrayDecayedType(dat->getElementType()),
|
|
|
|
dat->getSizeExpr(),
|
|
|
|
dat->getSizeModifier(),
|
|
|
|
dat->getIndexTypeCVRQualifiers(),
|
|
|
|
dat->getBracketsRange());
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Turn incomplete types into [*] types.
|
|
|
|
case Type::IncompleteArray: {
|
|
|
|
const IncompleteArrayType *iat = cast<IncompleteArrayType>(ty);
|
|
|
|
result = getVariableArrayType(
|
|
|
|
getVariableArrayDecayedType(iat->getElementType()),
|
|
|
|
/*size*/ 0,
|
|
|
|
ArrayType::Normal,
|
|
|
|
iat->getIndexTypeCVRQualifiers(),
|
|
|
|
SourceRange());
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Turn VLA types into [*] types.
|
|
|
|
case Type::VariableArray: {
|
|
|
|
const VariableArrayType *vat = cast<VariableArrayType>(ty);
|
|
|
|
result = getVariableArrayType(
|
|
|
|
getVariableArrayDecayedType(vat->getElementType()),
|
|
|
|
/*size*/ 0,
|
|
|
|
ArrayType::Star,
|
|
|
|
vat->getIndexTypeCVRQualifiers(),
|
|
|
|
vat->getBracketsRange());
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Apply the top-level qualifiers from the original.
|
2012-02-08 04:46:36 +04:00
|
|
|
return getQualifiedType(result, split.Quals);
|
2011-01-18 11:40:38 +03:00
|
|
|
}
|
2010-09-24 21:30:16 +04:00
|
|
|
|
2007-08-30 22:14:25 +04:00
|
|
|
/// getVariableArrayType - Returns a non-unique reference to the type for a
|
|
|
|
/// variable array of the specified element type.
|
2009-07-06 19:59:29 +04:00
|
|
|
QualType ASTContext::getVariableArrayType(QualType EltTy,
|
|
|
|
Expr *NumElts,
|
2007-08-30 22:10:14 +04:00
|
|
|
ArrayType::ArraySizeModifier ASM,
|
2011-01-27 22:55:10 +03:00
|
|
|
unsigned IndexTypeQuals,
|
2011-01-12 12:06:06 +03:00
|
|
|
SourceRange Brackets) const {
|
2008-02-15 21:16:39 +03:00
|
|
|
// Since we don't unique expressions, it isn't possible to unique VLA's
|
|
|
|
// that have an expression provided for their size.
|
2011-01-19 13:06:00 +03:00
|
|
|
QualType Canon;
|
2010-05-23 20:10:32 +04:00
|
|
|
|
2011-01-19 13:06:00 +03:00
|
|
|
// Be sure to pull qualifiers off the element type.
|
|
|
|
if (!EltTy.isCanonical() || EltTy.hasLocalQualifiers()) {
|
|
|
|
SplitQualType canonSplit = getCanonicalType(EltTy).split();
|
2012-02-08 04:46:36 +04:00
|
|
|
Canon = getVariableArrayType(QualType(canonSplit.Ty, 0), NumElts, ASM,
|
2011-01-27 22:55:10 +03:00
|
|
|
IndexTypeQuals, Brackets);
|
2012-02-08 04:46:36 +04:00
|
|
|
Canon = getQualifiedType(Canon, canonSplit.Quals);
|
2010-05-23 20:10:32 +04:00
|
|
|
}
|
|
|
|
|
2009-09-25 03:30:46 +04:00
|
|
|
VariableArrayType *New = new(*this, TypeAlignment)
|
2011-01-27 22:55:10 +03:00
|
|
|
VariableArrayType(EltTy, Canon, NumElts, ASM, IndexTypeQuals, Brackets);
|
2008-02-15 21:16:39 +03:00
|
|
|
|
|
|
|
VariableArrayTypes.push_back(New);
|
|
|
|
Types.push_back(New);
|
|
|
|
return QualType(New, 0);
|
|
|
|
}
|
|
|
|
|
2008-12-06 02:32:09 +03:00
|
|
|
/// getDependentSizedArrayType - Returns a non-unique reference to
|
|
|
|
/// the type for a dependently-sized array of the specified element
|
2009-07-31 04:23:35 +04:00
|
|
|
/// type.
|
2011-01-19 13:06:00 +03:00
|
|
|
QualType ASTContext::getDependentSizedArrayType(QualType elementType,
|
|
|
|
Expr *numElements,
|
2008-12-06 02:32:09 +03:00
|
|
|
ArrayType::ArraySizeModifier ASM,
|
2011-01-19 13:06:00 +03:00
|
|
|
unsigned elementTypeQuals,
|
|
|
|
SourceRange brackets) const {
|
|
|
|
assert((!numElements || numElements->isTypeDependent() ||
|
|
|
|
numElements->isValueDependent()) &&
|
2008-12-06 02:32:09 +03:00
|
|
|
"Size must be type- or value-dependent!");
|
|
|
|
|
2011-01-19 13:06:00 +03:00
|
|
|
// Dependently-sized array types that do not have a specified number
|
|
|
|
// of elements will have their sizes deduced from a dependent
|
|
|
|
// initializer. We do no canonicalization here at all, which is okay
|
|
|
|
// because they can't be used in most locations.
|
|
|
|
if (!numElements) {
|
|
|
|
DependentSizedArrayType *newType
|
|
|
|
= new (*this, TypeAlignment)
|
|
|
|
DependentSizedArrayType(*this, elementType, QualType(),
|
|
|
|
numElements, ASM, elementTypeQuals,
|
|
|
|
brackets);
|
|
|
|
Types.push_back(newType);
|
|
|
|
return QualType(newType, 0);
|
2009-11-19 21:03:26 +03:00
|
|
|
}
|
|
|
|
|
2011-01-19 13:06:00 +03:00
|
|
|
// Otherwise, we actually build a new type every time, but we
|
|
|
|
// also build a canonical type.
|
2009-09-09 19:08:12 +04:00
|
|
|
|
2011-01-19 13:06:00 +03:00
|
|
|
SplitQualType canonElementType = getCanonicalType(elementType).split();
|
2008-12-06 02:32:09 +03:00
|
|
|
|
2011-01-19 13:06:00 +03:00
|
|
|
void *insertPos = 0;
|
|
|
|
llvm::FoldingSetNodeID ID;
|
|
|
|
DependentSizedArrayType::Profile(ID, *this,
|
2012-02-08 04:46:36 +04:00
|
|
|
QualType(canonElementType.Ty, 0),
|
2011-01-19 13:06:00 +03:00
|
|
|
ASM, elementTypeQuals, numElements);
|
|
|
|
|
|
|
|
// Look for an existing type with these properties.
|
|
|
|
DependentSizedArrayType *canonTy =
|
|
|
|
DependentSizedArrayTypes.FindNodeOrInsertPos(ID, insertPos);
|
|
|
|
|
|
|
|
// If we don't have one, build one.
|
|
|
|
if (!canonTy) {
|
|
|
|
canonTy = new (*this, TypeAlignment)
|
2012-02-08 04:46:36 +04:00
|
|
|
DependentSizedArrayType(*this, QualType(canonElementType.Ty, 0),
|
2011-01-19 13:06:00 +03:00
|
|
|
QualType(), numElements, ASM, elementTypeQuals,
|
|
|
|
brackets);
|
|
|
|
DependentSizedArrayTypes.InsertNode(canonTy, insertPos);
|
|
|
|
Types.push_back(canonTy);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Apply qualifiers from the element type to the array.
|
|
|
|
QualType canon = getQualifiedType(QualType(canonTy,0),
|
2012-02-08 04:46:36 +04:00
|
|
|
canonElementType.Quals);
|
2011-01-19 13:06:00 +03:00
|
|
|
|
|
|
|
// If we didn't need extra canonicalization for the element type,
|
|
|
|
// then just use that as our result.
|
2012-02-08 04:46:36 +04:00
|
|
|
if (QualType(canonElementType.Ty, 0) == elementType)
|
2011-01-19 13:06:00 +03:00
|
|
|
return canon;
|
|
|
|
|
|
|
|
// Otherwise, we need to build a type which follows the spelling
|
|
|
|
// of the element type.
|
|
|
|
DependentSizedArrayType *sugaredType
|
|
|
|
= new (*this, TypeAlignment)
|
|
|
|
DependentSizedArrayType(*this, elementType, canon, numElements,
|
|
|
|
ASM, elementTypeQuals, brackets);
|
|
|
|
Types.push_back(sugaredType);
|
|
|
|
return QualType(sugaredType, 0);
|
|
|
|
}
|
|
|
|
|
|
|
|
QualType ASTContext::getIncompleteArrayType(QualType elementType,
|
2008-02-15 21:16:39 +03:00
|
|
|
ArrayType::ArraySizeModifier ASM,
|
2011-01-19 13:06:00 +03:00
|
|
|
unsigned elementTypeQuals) const {
|
2008-02-15 21:16:39 +03:00
|
|
|
llvm::FoldingSetNodeID ID;
|
2011-01-19 13:06:00 +03:00
|
|
|
IncompleteArrayType::Profile(ID, elementType, ASM, elementTypeQuals);
|
2008-02-15 21:16:39 +03:00
|
|
|
|
2011-01-19 13:06:00 +03:00
|
|
|
void *insertPos = 0;
|
|
|
|
if (IncompleteArrayType *iat =
|
|
|
|
IncompleteArrayTypes.FindNodeOrInsertPos(ID, insertPos))
|
|
|
|
return QualType(iat, 0);
|
2008-02-15 21:16:39 +03:00
|
|
|
|
|
|
|
// If the element type isn't canonical, this won't be a canonical type
|
2011-01-19 13:06:00 +03:00
|
|
|
// either, so fill in the canonical type field. We also have to pull
|
|
|
|
// qualifiers off the element type.
|
|
|
|
QualType canon;
|
2008-02-15 21:16:39 +03:00
|
|
|
|
2011-01-19 13:06:00 +03:00
|
|
|
if (!elementType.isCanonical() || elementType.hasLocalQualifiers()) {
|
|
|
|
SplitQualType canonSplit = getCanonicalType(elementType).split();
|
2012-02-08 04:46:36 +04:00
|
|
|
canon = getIncompleteArrayType(QualType(canonSplit.Ty, 0),
|
2011-01-19 13:06:00 +03:00
|
|
|
ASM, elementTypeQuals);
|
2012-02-08 04:46:36 +04:00
|
|
|
canon = getQualifiedType(canon, canonSplit.Quals);
|
2008-02-15 21:16:39 +03:00
|
|
|
|
|
|
|
// Get the new insert position for the node we care about.
|
2011-01-19 13:06:00 +03:00
|
|
|
IncompleteArrayType *existing =
|
|
|
|
IncompleteArrayTypes.FindNodeOrInsertPos(ID, insertPos);
|
|
|
|
assert(!existing && "Shouldn't be in the map!"); (void) existing;
|
2007-10-30 02:37:31 +03:00
|
|
|
}
|
2008-02-15 21:16:39 +03:00
|
|
|
|
2011-01-19 13:06:00 +03:00
|
|
|
IncompleteArrayType *newType = new (*this, TypeAlignment)
|
|
|
|
IncompleteArrayType(elementType, canon, ASM, elementTypeQuals);
|
2008-02-15 21:16:39 +03:00
|
|
|
|
2011-01-19 13:06:00 +03:00
|
|
|
IncompleteArrayTypes.InsertNode(newType, insertPos);
|
|
|
|
Types.push_back(newType);
|
|
|
|
return QualType(newType, 0);
|
2007-08-30 05:06:46 +04:00
|
|
|
}
|
|
|
|
|
2007-07-18 22:00:27 +04:00
|
|
|
/// getVectorType - Return the unique reference to a vector type of
|
|
|
|
/// the specified element type and size. VectorType must be a built-in type.
|
2010-02-05 03:12:22 +03:00
|
|
|
QualType ASTContext::getVectorType(QualType vecType, unsigned NumElts,
|
2011-01-12 12:06:06 +03:00
|
|
|
VectorType::VectorKind VecKind) const {
|
2011-01-19 13:06:00 +03:00
|
|
|
assert(vecType->isBuiltinType());
|
2009-09-09 19:08:12 +04:00
|
|
|
|
2007-07-11 21:01:13 +04:00
|
|
|
// Check if we've already instantiated a vector of this type.
|
|
|
|
llvm::FoldingSetNodeID ID;
|
2010-11-11 00:56:12 +03:00
|
|
|
VectorType::Profile(ID, vecType, NumElts, Type::Vector, VecKind);
|
2010-06-23 10:00:24 +04:00
|
|
|
|
2007-07-11 21:01:13 +04:00
|
|
|
void *InsertPos = 0;
|
|
|
|
if (VectorType *VTP = VectorTypes.FindNodeOrInsertPos(ID, InsertPos))
|
|
|
|
return QualType(VTP, 0);
|
|
|
|
|
|
|
|
// If the element type isn't canonical, this won't be a canonical type either,
|
|
|
|
// so fill in the canonical type field.
|
|
|
|
QualType Canonical;
|
2010-08-06 14:14:59 +04:00
|
|
|
if (!vecType.isCanonical()) {
|
2010-11-16 03:32:20 +03:00
|
|
|
Canonical = getVectorType(getCanonicalType(vecType), NumElts, VecKind);
|
2009-09-09 19:08:12 +04:00
|
|
|
|
2007-07-11 21:01:13 +04:00
|
|
|
// Get the new insert position for the node we care about.
|
|
|
|
VectorType *NewIP = VectorTypes.FindNodeOrInsertPos(ID, InsertPos);
|
2010-12-23 04:01:28 +03:00
|
|
|
assert(NewIP == 0 && "Shouldn't be in the map!"); (void)NewIP;
|
2007-07-11 21:01:13 +04:00
|
|
|
}
|
2009-09-25 03:30:46 +04:00
|
|
|
VectorType *New = new (*this, TypeAlignment)
|
2010-11-11 00:56:12 +03:00
|
|
|
VectorType(vecType, NumElts, Canonical, VecKind);
|
2007-07-11 21:01:13 +04:00
|
|
|
VectorTypes.InsertNode(New, InsertPos);
|
|
|
|
Types.push_back(New);
|
|
|
|
return QualType(New, 0);
|
|
|
|
}
|
|
|
|
|
2008-04-19 03:10:10 +04:00
|
|
|
/// getExtVectorType - Return the unique reference to an extended vector type of
|
2007-07-18 22:00:27 +04:00
|
|
|
/// the specified element type and size. VectorType must be a built-in type.
|
2011-01-12 12:06:06 +03:00
|
|
|
QualType
|
|
|
|
ASTContext::getExtVectorType(QualType vecType, unsigned NumElts) const {
|
2011-06-15 20:02:29 +04:00
|
|
|
assert(vecType->isBuiltinType() || vecType->isDependentType());
|
2009-09-09 19:08:12 +04:00
|
|
|
|
2007-07-18 22:00:27 +04:00
|
|
|
// Check if we've already instantiated a vector of this type.
|
|
|
|
llvm::FoldingSetNodeID ID;
|
2010-06-23 10:00:24 +04:00
|
|
|
VectorType::Profile(ID, vecType, NumElts, Type::ExtVector,
|
2010-11-11 00:56:12 +03:00
|
|
|
VectorType::GenericVector);
|
2007-07-18 22:00:27 +04:00
|
|
|
void *InsertPos = 0;
|
|
|
|
if (VectorType *VTP = VectorTypes.FindNodeOrInsertPos(ID, InsertPos))
|
|
|
|
return QualType(VTP, 0);
|
|
|
|
|
|
|
|
// If the element type isn't canonical, this won't be a canonical type either,
|
|
|
|
// so fill in the canonical type field.
|
|
|
|
QualType Canonical;
|
2009-10-23 00:10:53 +04:00
|
|
|
if (!vecType.isCanonical()) {
|
2008-04-19 03:10:10 +04:00
|
|
|
Canonical = getExtVectorType(getCanonicalType(vecType), NumElts);
|
2009-09-09 19:08:12 +04:00
|
|
|
|
2007-07-18 22:00:27 +04:00
|
|
|
// Get the new insert position for the node we care about.
|
|
|
|
VectorType *NewIP = VectorTypes.FindNodeOrInsertPos(ID, InsertPos);
|
2010-12-23 04:01:28 +03:00
|
|
|
assert(NewIP == 0 && "Shouldn't be in the map!"); (void)NewIP;
|
2007-07-18 22:00:27 +04:00
|
|
|
}
|
2009-09-25 03:30:46 +04:00
|
|
|
ExtVectorType *New = new (*this, TypeAlignment)
|
|
|
|
ExtVectorType(vecType, NumElts, Canonical);
|
2007-07-18 22:00:27 +04:00
|
|
|
VectorTypes.InsertNode(New, InsertPos);
|
|
|
|
Types.push_back(New);
|
|
|
|
return QualType(New, 0);
|
|
|
|
}
|
|
|
|
|
2011-01-12 12:06:06 +03:00
|
|
|
QualType
|
|
|
|
ASTContext::getDependentSizedExtVectorType(QualType vecType,
|
|
|
|
Expr *SizeExpr,
|
|
|
|
SourceLocation AttrLoc) const {
|
2009-07-31 07:54:25 +04:00
|
|
|
llvm::FoldingSetNodeID ID;
|
2009-09-09 19:08:12 +04:00
|
|
|
DependentSizedExtVectorType::Profile(ID, *this, getCanonicalType(vecType),
|
2009-07-31 07:54:25 +04:00
|
|
|
SizeExpr);
|
2009-09-09 19:08:12 +04:00
|
|
|
|
2009-07-31 07:54:25 +04:00
|
|
|
void *InsertPos = 0;
|
|
|
|
DependentSizedExtVectorType *Canon
|
|
|
|
= DependentSizedExtVectorTypes.FindNodeOrInsertPos(ID, InsertPos);
|
|
|
|
DependentSizedExtVectorType *New;
|
|
|
|
if (Canon) {
|
|
|
|
// We already have a canonical version of this array type; use it as
|
|
|
|
// the canonical type for a newly-built type.
|
2009-09-25 03:30:46 +04:00
|
|
|
New = new (*this, TypeAlignment)
|
|
|
|
DependentSizedExtVectorType(*this, vecType, QualType(Canon, 0),
|
|
|
|
SizeExpr, AttrLoc);
|
2009-07-31 07:54:25 +04:00
|
|
|
} else {
|
|
|
|
QualType CanonVecTy = getCanonicalType(vecType);
|
|
|
|
if (CanonVecTy == vecType) {
|
2009-09-25 03:30:46 +04:00
|
|
|
New = new (*this, TypeAlignment)
|
|
|
|
DependentSizedExtVectorType(*this, vecType, QualType(), SizeExpr,
|
|
|
|
AttrLoc);
|
Fix an obscure crash found in the Boost.MPL test suite, along with a
ton of potential crashes of the same kind. The fundamental problem is
that type creation was following a dangerous pattern when using its
FoldingSets:
1) Use FindNodeOrInsertPos to see if the type is available
2) If not, and we aren't looking at a canonical type, build the
canonical type
3) Build and insert the new node into the FoldingSet
The problem here is that building the canonical type can, in very rare
circumstances, force the hash table inside the FoldingSet to
reallocate. That invalidates the insertion position we computed in
step 1, and in step 3 we end up inserting the new node into the wrong
place. BOOM!
I've audited all of ASTContext, fixing this problem everywhere I found
it. The vast majority of wrong code was C++-specific (and *ahem*
written by me), so I also audited other major folding sets in the C++
code (e.g., template specializations), but found no other instances of
this problem.
git-svn-id: https://llvm.org/svn/llvm-project/cfe/trunk@95315 91177308-0d34-0410-b5e6-96231b3b80d8
2010-02-04 21:10:26 +03:00
|
|
|
|
|
|
|
DependentSizedExtVectorType *CanonCheck
|
|
|
|
= DependentSizedExtVectorTypes.FindNodeOrInsertPos(ID, InsertPos);
|
|
|
|
assert(!CanonCheck && "Dependent-sized ext_vector canonical type broken");
|
|
|
|
(void)CanonCheck;
|
2009-07-31 07:54:25 +04:00
|
|
|
DependentSizedExtVectorTypes.InsertNode(New, InsertPos);
|
|
|
|
} else {
|
|
|
|
QualType Canon = getDependentSizedExtVectorType(CanonVecTy, SizeExpr,
|
|
|
|
SourceLocation());
|
2009-09-25 03:30:46 +04:00
|
|
|
New = new (*this, TypeAlignment)
|
|
|
|
DependentSizedExtVectorType(*this, vecType, Canon, SizeExpr, AttrLoc);
|
2009-07-31 07:54:25 +04:00
|
|
|
}
|
|
|
|
}
|
2009-09-09 19:08:12 +04:00
|
|
|
|
2009-06-18 01:51:59 +04:00
|
|
|
Types.push_back(New);
|
|
|
|
return QualType(New, 0);
|
|
|
|
}
|
|
|
|
|
2009-02-27 02:50:07 +03:00
|
|
|
/// getFunctionNoProtoType - Return a K&R style C function type like 'int()'.
|
2007-07-11 21:01:13 +04:00
|
|
|
///
|
2011-01-12 12:06:06 +03:00
|
|
|
QualType
|
|
|
|
ASTContext::getFunctionNoProtoType(QualType ResultTy,
|
|
|
|
const FunctionType::ExtInfo &Info) const {
|
2011-03-01 20:40:53 +03:00
|
|
|
const CallingConv DefaultCC = Info.getCC();
|
|
|
|
const CallingConv CallConv = (LangOpts.MRTD && DefaultCC == CC_Default) ?
|
|
|
|
CC_X86StdCall : DefaultCC;
|
2007-07-11 21:01:13 +04:00
|
|
|
// Unique functions, to guarantee there is only one function of a particular
|
|
|
|
// structure.
|
|
|
|
llvm::FoldingSetNodeID ID;
|
2010-03-31 00:24:48 +04:00
|
|
|
FunctionNoProtoType::Profile(ID, ResultTy, Info);
|
2009-09-09 19:08:12 +04:00
|
|
|
|
2007-07-11 21:01:13 +04:00
|
|
|
void *InsertPos = 0;
|
2009-09-09 19:08:12 +04:00
|
|
|
if (FunctionNoProtoType *FT =
|
2009-02-27 02:50:07 +03:00
|
|
|
FunctionNoProtoTypes.FindNodeOrInsertPos(ID, InsertPos))
|
2007-07-11 21:01:13 +04:00
|
|
|
return QualType(FT, 0);
|
2009-09-09 19:08:12 +04:00
|
|
|
|
2007-07-11 21:01:13 +04:00
|
|
|
QualType Canonical;
|
2010-01-18 20:14:39 +03:00
|
|
|
if (!ResultTy.isCanonical() ||
|
2010-02-06 00:31:56 +03:00
|
|
|
getCanonicalCallConv(CallConv) != CallConv) {
|
2010-03-31 00:24:48 +04:00
|
|
|
Canonical =
|
|
|
|
getFunctionNoProtoType(getCanonicalType(ResultTy),
|
|
|
|
Info.withCallingConv(getCanonicalCallConv(CallConv)));
|
2009-09-09 19:08:12 +04:00
|
|
|
|
2007-07-11 21:01:13 +04:00
|
|
|
// Get the new insert position for the node we care about.
|
2009-02-27 02:50:07 +03:00
|
|
|
FunctionNoProtoType *NewIP =
|
|
|
|
FunctionNoProtoTypes.FindNodeOrInsertPos(ID, InsertPos);
|
2010-12-23 04:01:28 +03:00
|
|
|
assert(NewIP == 0 && "Shouldn't be in the map!"); (void)NewIP;
|
2007-07-11 21:01:13 +04:00
|
|
|
}
|
2009-09-09 19:08:12 +04:00
|
|
|
|
2011-03-01 20:40:53 +03:00
|
|
|
FunctionProtoType::ExtInfo newInfo = Info.withCallingConv(CallConv);
|
2009-09-25 03:30:46 +04:00
|
|
|
FunctionNoProtoType *New = new (*this, TypeAlignment)
|
2011-03-01 20:40:53 +03:00
|
|
|
FunctionNoProtoType(ResultTy, Canonical, newInfo);
|
2007-07-11 21:01:13 +04:00
|
|
|
Types.push_back(New);
|
2009-02-27 02:50:07 +03:00
|
|
|
FunctionNoProtoTypes.InsertNode(New, InsertPos);
|
2007-07-11 21:01:13 +04:00
|
|
|
return QualType(New, 0);
|
|
|
|
}
|
|
|
|
|
|
|
|
/// getFunctionType - Return a normal function type with a typed argument
|
|
|
|
/// list. isVariadic indicates whether the argument list includes '...'.
|
2011-01-12 12:06:06 +03:00
|
|
|
QualType
|
|
|
|
ASTContext::getFunctionType(QualType ResultTy,
|
|
|
|
const QualType *ArgArray, unsigned NumArgs,
|
|
|
|
const FunctionProtoType::ExtProtoInfo &EPI) const {
|
2007-07-11 21:01:13 +04:00
|
|
|
// Unique functions, to guarantee there is only one function of a particular
|
|
|
|
// structure.
|
|
|
|
llvm::FoldingSetNodeID ID;
|
2011-03-13 20:09:40 +03:00
|
|
|
FunctionProtoType::Profile(ID, ResultTy, ArgArray, NumArgs, EPI, *this);
|
2007-07-11 21:01:13 +04:00
|
|
|
|
|
|
|
void *InsertPos = 0;
|
2009-09-09 19:08:12 +04:00
|
|
|
if (FunctionProtoType *FTP =
|
2009-02-27 02:50:07 +03:00
|
|
|
FunctionProtoTypes.FindNodeOrInsertPos(ID, InsertPos))
|
2007-07-11 21:01:13 +04:00
|
|
|
return QualType(FTP, 0);
|
2009-05-28 02:11:52 +04:00
|
|
|
|
|
|
|
// Determine whether the type being created is already canonical or not.
|
2012-02-10 13:58:53 +04:00
|
|
|
bool isCanonical =
|
|
|
|
EPI.ExceptionSpecType == EST_None && ResultTy.isCanonical() &&
|
|
|
|
!EPI.HasTrailingReturn;
|
2007-07-11 21:01:13 +04:00
|
|
|
for (unsigned i = 0; i != NumArgs && isCanonical; ++i)
|
2009-10-23 02:37:11 +04:00
|
|
|
if (!ArgArray[i].isCanonicalAsParam())
|
2007-07-11 21:01:13 +04:00
|
|
|
isCanonical = false;
|
|
|
|
|
2011-03-01 20:40:53 +03:00
|
|
|
const CallingConv DefaultCC = EPI.ExtInfo.getCC();
|
|
|
|
const CallingConv CallConv = (LangOpts.MRTD && DefaultCC == CC_Default) ?
|
|
|
|
CC_X86StdCall : DefaultCC;
|
2010-12-14 11:05:40 +03:00
|
|
|
|
2007-07-11 21:01:13 +04:00
|
|
|
// If this type isn't canonical, get the canonical version of it.
|
2009-05-28 02:11:52 +04:00
|
|
|
// The exception spec is not part of the canonical type.
|
2007-07-11 21:01:13 +04:00
|
|
|
QualType Canonical;
|
2010-02-06 00:31:56 +03:00
|
|
|
if (!isCanonical || getCanonicalCallConv(CallConv) != CallConv) {
|
2011-07-23 14:55:15 +04:00
|
|
|
SmallVector<QualType, 16> CanonicalArgs;
|
2007-07-11 21:01:13 +04:00
|
|
|
CanonicalArgs.reserve(NumArgs);
|
|
|
|
for (unsigned i = 0; i != NumArgs; ++i)
|
2009-10-23 02:37:11 +04:00
|
|
|
CanonicalArgs.push_back(getCanonicalParamType(ArgArray[i]));
|
2009-05-28 02:11:52 +04:00
|
|
|
|
2010-12-14 11:05:40 +03:00
|
|
|
FunctionProtoType::ExtProtoInfo CanonicalEPI = EPI;
|
2012-02-10 13:58:53 +04:00
|
|
|
CanonicalEPI.HasTrailingReturn = false;
|
2011-03-06 13:52:04 +03:00
|
|
|
CanonicalEPI.ExceptionSpecType = EST_None;
|
|
|
|
CanonicalEPI.NumExceptions = 0;
|
2010-12-14 11:05:40 +03:00
|
|
|
CanonicalEPI.ExtInfo
|
|
|
|
= CanonicalEPI.ExtInfo.withCallingConv(getCanonicalCallConv(CallConv));
|
|
|
|
|
2008-04-07 02:59:24 +04:00
|
|
|
Canonical = getFunctionType(getCanonicalType(ResultTy),
|
2009-05-21 13:52:38 +04:00
|
|
|
CanonicalArgs.data(), NumArgs,
|
2010-12-14 11:05:40 +03:00
|
|
|
CanonicalEPI);
|
2009-05-28 02:11:52 +04:00
|
|
|
|
2007-07-11 21:01:13 +04:00
|
|
|
// Get the new insert position for the node we care about.
|
2009-02-27 02:50:07 +03:00
|
|
|
FunctionProtoType *NewIP =
|
|
|
|
FunctionProtoTypes.FindNodeOrInsertPos(ID, InsertPos);
|
2010-12-23 04:01:28 +03:00
|
|
|
assert(NewIP == 0 && "Shouldn't be in the map!"); (void)NewIP;
|
2007-07-11 21:01:13 +04:00
|
|
|
}
|
2009-05-28 02:11:52 +04:00
|
|
|
|
2011-06-16 03:02:42 +04:00
|
|
|
// FunctionProtoType objects are allocated with extra bytes after
|
|
|
|
// them for three variable size arrays at the end:
|
|
|
|
// - parameter types
|
|
|
|
// - exception types
|
|
|
|
// - consumed-arguments flags
|
|
|
|
// Instead of the exception types, there could be a noexcept
|
|
|
|
// expression.
|
2010-12-14 11:05:40 +03:00
|
|
|
size_t Size = sizeof(FunctionProtoType) +
|
2011-03-12 14:50:43 +03:00
|
|
|
NumArgs * sizeof(QualType);
|
|
|
|
if (EPI.ExceptionSpecType == EST_Dynamic)
|
|
|
|
Size += EPI.NumExceptions * sizeof(QualType);
|
|
|
|
else if (EPI.ExceptionSpecType == EST_ComputedNoexcept) {
|
2011-03-13 20:09:40 +03:00
|
|
|
Size += sizeof(Expr*);
|
2011-03-12 14:50:43 +03:00
|
|
|
}
|
2011-06-16 03:02:42 +04:00
|
|
|
if (EPI.ConsumedArguments)
|
|
|
|
Size += NumArgs * sizeof(bool);
|
|
|
|
|
2010-12-14 11:05:40 +03:00
|
|
|
FunctionProtoType *FTP = (FunctionProtoType*) Allocate(Size, TypeAlignment);
|
2011-03-01 20:40:53 +03:00
|
|
|
FunctionProtoType::ExtProtoInfo newEPI = EPI;
|
|
|
|
newEPI.ExtInfo = EPI.ExtInfo.withCallingConv(CallConv);
|
2011-03-13 20:09:40 +03:00
|
|
|
new (FTP) FunctionProtoType(ResultTy, ArgArray, NumArgs, Canonical, newEPI);
|
2007-07-11 21:01:13 +04:00
|
|
|
Types.push_back(FTP);
|
2009-02-27 02:50:07 +03:00
|
|
|
FunctionProtoTypes.InsertNode(FTP, InsertPos);
|
2007-07-11 21:01:13 +04:00
|
|
|
return QualType(FTP, 0);
|
|
|
|
}
|
|
|
|
|
2010-03-10 06:28:59 +03:00
|
|
|
#ifndef NDEBUG
|
|
|
|
static bool NeedsInjectedClassNameType(const RecordDecl *D) {
|
|
|
|
if (!isa<CXXRecordDecl>(D)) return false;
|
|
|
|
const CXXRecordDecl *RD = cast<CXXRecordDecl>(D);
|
|
|
|
if (isa<ClassTemplatePartialSpecializationDecl>(RD))
|
|
|
|
return true;
|
|
|
|
if (RD->getDescribedClassTemplate() &&
|
|
|
|
!isa<ClassTemplateSpecializationDecl>(RD))
|
|
|
|
return true;
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
/// getInjectedClassNameType - Return the unique reference to the
|
|
|
|
/// injected class name type for the specified templated declaration.
|
|
|
|
QualType ASTContext::getInjectedClassNameType(CXXRecordDecl *Decl,
|
2011-01-12 12:06:06 +03:00
|
|
|
QualType TST) const {
|
2010-03-10 06:28:59 +03:00
|
|
|
assert(NeedsInjectedClassNameType(Decl));
|
|
|
|
if (Decl->TypeForDecl) {
|
|
|
|
assert(isa<InjectedClassNameType>(Decl->TypeForDecl));
|
2012-01-14 20:38:05 +04:00
|
|
|
} else if (CXXRecordDecl *PrevDecl = Decl->getPreviousDecl()) {
|
2010-03-10 06:28:59 +03:00
|
|
|
assert(PrevDecl->TypeForDecl && "previous declaration has no type");
|
|
|
|
Decl->TypeForDecl = PrevDecl->TypeForDecl;
|
|
|
|
assert(isa<InjectedClassNameType>(Decl->TypeForDecl));
|
|
|
|
} else {
|
2011-01-19 09:33:43 +03:00
|
|
|
Type *newType =
|
2010-04-27 04:57:59 +04:00
|
|
|
new (*this, TypeAlignment) InjectedClassNameType(Decl, TST);
|
2011-01-19 09:33:43 +03:00
|
|
|
Decl->TypeForDecl = newType;
|
|
|
|
Types.push_back(newType);
|
2010-03-10 06:28:59 +03:00
|
|
|
}
|
|
|
|
return QualType(Decl->TypeForDecl, 0);
|
|
|
|
}
|
|
|
|
|
2008-04-14 01:07:44 +04:00
|
|
|
/// getTypeDeclType - Return the unique reference to the type for the
|
|
|
|
/// specified type declaration.
|
2011-01-12 12:06:06 +03:00
|
|
|
QualType ASTContext::getTypeDeclTypeSlow(const TypeDecl *Decl) const {
|
2008-10-16 20:50:47 +04:00
|
|
|
assert(Decl && "Passed null for Decl param");
|
2010-03-10 09:48:02 +03:00
|
|
|
assert(!Decl->TypeForDecl && "TypeForDecl present in slow case");
|
2009-09-09 19:08:12 +04:00
|
|
|
|
2011-04-15 18:24:37 +04:00
|
|
|
if (const TypedefNameDecl *Typedef = dyn_cast<TypedefNameDecl>(Decl))
|
2008-04-14 01:07:44 +04:00
|
|
|
return getTypedefType(Typedef);
|
2010-03-10 09:48:02 +03:00
|
|
|
|
|
|
|
assert(!isa<TemplateTypeParmDecl>(Decl) &&
|
|
|
|
"Template type parameter types are always available.");
|
|
|
|
|
2010-02-16 06:57:14 +03:00
|
|
|
if (const RecordDecl *Record = dyn_cast<RecordDecl>(Decl)) {
|
2012-01-14 20:38:05 +04:00
|
|
|
assert(!Record->getPreviousDecl() &&
|
2010-03-10 09:48:02 +03:00
|
|
|
"struct/union has previous declaration");
|
|
|
|
assert(!NeedsInjectedClassNameType(Record));
|
2010-07-05 01:44:47 +04:00
|
|
|
return getRecordType(Record);
|
2010-02-16 06:57:14 +03:00
|
|
|
} else if (const EnumDecl *Enum = dyn_cast<EnumDecl>(Decl)) {
|
2012-01-14 20:38:05 +04:00
|
|
|
assert(!Enum->getPreviousDecl() &&
|
2010-03-10 09:48:02 +03:00
|
|
|
"enum has previous declaration");
|
2010-07-05 01:44:47 +04:00
|
|
|
return getEnumType(Enum);
|
2010-02-16 06:57:14 +03:00
|
|
|
} else if (const UnresolvedUsingTypenameDecl *Using =
|
2009-12-05 01:46:56 +03:00
|
|
|
dyn_cast<UnresolvedUsingTypenameDecl>(Decl)) {
|
2011-01-19 09:33:43 +03:00
|
|
|
Type *newType = new (*this, TypeAlignment) UnresolvedUsingType(Using);
|
|
|
|
Decl->TypeForDecl = newType;
|
|
|
|
Types.push_back(newType);
|
2009-07-31 06:02:20 +04:00
|
|
|
} else
|
2010-03-10 09:48:02 +03:00
|
|
|
llvm_unreachable("TypeDecl without a type?");
|
2008-08-08 00:55:28 +04:00
|
|
|
|
|
|
|
return QualType(Decl->TypeForDecl, 0);
|
2008-04-14 01:07:44 +04:00
|
|
|
}
|
|
|
|
|
2007-07-11 21:01:13 +04:00
|
|
|
/// getTypedefType - Return the unique reference to the type for the
|
2011-04-15 18:24:37 +04:00
|
|
|
/// specified typedef name decl.
|
2010-07-02 15:55:11 +04:00
|
|
|
QualType
|
2011-04-15 18:24:37 +04:00
|
|
|
ASTContext::getTypedefType(const TypedefNameDecl *Decl,
|
|
|
|
QualType Canonical) const {
|
2007-07-11 21:01:13 +04:00
|
|
|
if (Decl->TypeForDecl) return QualType(Decl->TypeForDecl, 0);
|
2009-09-09 19:08:12 +04:00
|
|
|
|
2010-07-02 15:55:11 +04:00
|
|
|
if (Canonical.isNull())
|
|
|
|
Canonical = getCanonicalType(Decl->getUnderlyingType());
|
2011-01-19 09:33:43 +03:00
|
|
|
TypedefType *newType = new(*this, TypeAlignment)
|
2009-09-25 03:30:46 +04:00
|
|
|
TypedefType(Type::Typedef, Decl, Canonical);
|
2011-01-19 09:33:43 +03:00
|
|
|
Decl->TypeForDecl = newType;
|
|
|
|
Types.push_back(newType);
|
|
|
|
return QualType(newType, 0);
|
2007-07-11 21:01:13 +04:00
|
|
|
}
|
|
|
|
|
2011-01-12 12:06:06 +03:00
|
|
|
QualType ASTContext::getRecordType(const RecordDecl *Decl) const {
|
2010-07-05 01:44:47 +04:00
|
|
|
if (Decl->TypeForDecl) return QualType(Decl->TypeForDecl, 0);
|
|
|
|
|
2012-01-14 20:38:05 +04:00
|
|
|
if (const RecordDecl *PrevDecl = Decl->getPreviousDecl())
|
2010-07-05 01:44:47 +04:00
|
|
|
if (PrevDecl->TypeForDecl)
|
|
|
|
return QualType(Decl->TypeForDecl = PrevDecl->TypeForDecl, 0);
|
|
|
|
|
2011-01-19 09:33:43 +03:00
|
|
|
RecordType *newType = new (*this, TypeAlignment) RecordType(Decl);
|
|
|
|
Decl->TypeForDecl = newType;
|
|
|
|
Types.push_back(newType);
|
|
|
|
return QualType(newType, 0);
|
2010-07-05 01:44:47 +04:00
|
|
|
}
|
|
|
|
|
2011-01-12 12:06:06 +03:00
|
|
|
QualType ASTContext::getEnumType(const EnumDecl *Decl) const {
|
2010-07-05 01:44:47 +04:00
|
|
|
if (Decl->TypeForDecl) return QualType(Decl->TypeForDecl, 0);
|
|
|
|
|
2012-01-14 20:38:05 +04:00
|
|
|
if (const EnumDecl *PrevDecl = Decl->getPreviousDecl())
|
2010-07-05 01:44:47 +04:00
|
|
|
if (PrevDecl->TypeForDecl)
|
|
|
|
return QualType(Decl->TypeForDecl = PrevDecl->TypeForDecl, 0);
|
|
|
|
|
2011-01-19 09:33:43 +03:00
|
|
|
EnumType *newType = new (*this, TypeAlignment) EnumType(Decl);
|
|
|
|
Decl->TypeForDecl = newType;
|
|
|
|
Types.push_back(newType);
|
|
|
|
return QualType(newType, 0);
|
2010-07-05 01:44:47 +04:00
|
|
|
}
|
|
|
|
|
2011-01-06 04:58:22 +03:00
|
|
|
QualType ASTContext::getAttributedType(AttributedType::Kind attrKind,
|
|
|
|
QualType modifiedType,
|
|
|
|
QualType equivalentType) {
|
|
|
|
llvm::FoldingSetNodeID id;
|
|
|
|
AttributedType::Profile(id, attrKind, modifiedType, equivalentType);
|
|
|
|
|
|
|
|
void *insertPos = 0;
|
|
|
|
AttributedType *type = AttributedTypes.FindNodeOrInsertPos(id, insertPos);
|
|
|
|
if (type) return QualType(type, 0);
|
|
|
|
|
|
|
|
QualType canon = getCanonicalType(equivalentType);
|
|
|
|
type = new (*this, TypeAlignment)
|
|
|
|
AttributedType(canon, attrKind, modifiedType, equivalentType);
|
|
|
|
|
|
|
|
Types.push_back(type);
|
|
|
|
AttributedTypes.InsertNode(type, insertPos);
|
|
|
|
|
|
|
|
return QualType(type, 0);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2009-10-18 13:09:24 +04:00
|
|
|
/// \brief Retrieve a substitution-result type.
|
|
|
|
QualType
|
|
|
|
ASTContext::getSubstTemplateTypeParmType(const TemplateTypeParmType *Parm,
|
2011-01-12 12:06:06 +03:00
|
|
|
QualType Replacement) const {
|
2009-10-23 00:10:53 +04:00
|
|
|
assert(Replacement.isCanonical()
|
2009-10-18 13:09:24 +04:00
|
|
|
&& "replacement types must always be canonical");
|
|
|
|
|
|
|
|
llvm::FoldingSetNodeID ID;
|
|
|
|
SubstTemplateTypeParmType::Profile(ID, Parm, Replacement);
|
|
|
|
void *InsertPos = 0;
|
|
|
|
SubstTemplateTypeParmType *SubstParm
|
|
|
|
= SubstTemplateTypeParmTypes.FindNodeOrInsertPos(ID, InsertPos);
|
|
|
|
|
|
|
|
if (!SubstParm) {
|
|
|
|
SubstParm = new (*this, TypeAlignment)
|
|
|
|
SubstTemplateTypeParmType(Parm, Replacement);
|
|
|
|
Types.push_back(SubstParm);
|
|
|
|
SubstTemplateTypeParmTypes.InsertNode(SubstParm, InsertPos);
|
|
|
|
}
|
|
|
|
|
|
|
|
return QualType(SubstParm, 0);
|
|
|
|
}
|
|
|
|
|
2011-01-14 05:55:32 +03:00
|
|
|
/// \brief Retrieve a
|
|
|
|
QualType ASTContext::getSubstTemplateTypeParmPackType(
|
|
|
|
const TemplateTypeParmType *Parm,
|
|
|
|
const TemplateArgument &ArgPack) {
|
|
|
|
#ifndef NDEBUG
|
|
|
|
for (TemplateArgument::pack_iterator P = ArgPack.pack_begin(),
|
|
|
|
PEnd = ArgPack.pack_end();
|
|
|
|
P != PEnd; ++P) {
|
|
|
|
assert(P->getKind() == TemplateArgument::Type &&"Pack contains a non-type");
|
|
|
|
assert(P->getAsType().isCanonical() && "Pack contains non-canonical type");
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
llvm::FoldingSetNodeID ID;
|
|
|
|
SubstTemplateTypeParmPackType::Profile(ID, Parm, ArgPack);
|
|
|
|
void *InsertPos = 0;
|
|
|
|
if (SubstTemplateTypeParmPackType *SubstParm
|
|
|
|
= SubstTemplateTypeParmPackTypes.FindNodeOrInsertPos(ID, InsertPos))
|
|
|
|
return QualType(SubstParm, 0);
|
|
|
|
|
|
|
|
QualType Canon;
|
|
|
|
if (!Parm->isCanonicalUnqualified()) {
|
|
|
|
Canon = getCanonicalType(QualType(Parm, 0));
|
|
|
|
Canon = getSubstTemplateTypeParmPackType(cast<TemplateTypeParmType>(Canon),
|
|
|
|
ArgPack);
|
|
|
|
SubstTemplateTypeParmPackTypes.FindNodeOrInsertPos(ID, InsertPos);
|
|
|
|
}
|
|
|
|
|
|
|
|
SubstTemplateTypeParmPackType *SubstParm
|
|
|
|
= new (*this, TypeAlignment) SubstTemplateTypeParmPackType(Parm, Canon,
|
|
|
|
ArgPack);
|
|
|
|
Types.push_back(SubstParm);
|
|
|
|
SubstTemplateTypeParmTypes.InsertNode(SubstParm, InsertPos);
|
|
|
|
return QualType(SubstParm, 0);
|
|
|
|
}
|
|
|
|
|
2009-02-06 02:33:38 +03:00
|
|
|
/// \brief Retrieve the template type parameter type for a template
|
2009-09-09 19:08:12 +04:00
|
|
|
/// parameter or parameter pack with the given depth, index, and (optionally)
|
2009-06-16 04:30:48 +04:00
|
|
|
/// name.
|
2009-09-09 19:08:12 +04:00
|
|
|
QualType ASTContext::getTemplateTypeParmType(unsigned Depth, unsigned Index,
|
2009-06-16 04:30:48 +04:00
|
|
|
bool ParameterPack,
|
Re-applies the patch first applied way back in r106099, with
accompanying fixes to make it work today.
The core of this patch is to provide a link from a TemplateTypeParmType
back to the TemplateTypeParmDecl node which declared it. This in turn
provides much more precise information about the type, where it came
from, and how it functions for AST consumers.
To make the patch work almost a year after its first attempt, it needed
serialization support, and it now retains the old getName() interface.
Finally, it requires us to not attempt to instantiate the type in an
unsupported friend decl -- specifically those coming from template
friend decls but which refer to a specific type through a dependent
name.
A cleaner representation of the last item would be to build
FriendTemplateDecl nodes for these, storing their template parameters
etc, and to perform proper instantation of them like any other template
declaration. They can still be flagged as unsupported for the purpose of
access checking, etc.
This passed an asserts-enabled bootstrap for me, and the reduced test
case mentioned in the original review thread no longer causes issues,
likely fixed at somewhere amidst the 24k revisions that have elapsed.
git-svn-id: https://llvm.org/svn/llvm-project/cfe/trunk@130628 91177308-0d34-0410-b5e6-96231b3b80d8
2011-05-01 04:51:33 +04:00
|
|
|
TemplateTypeParmDecl *TTPDecl) const {
|
2009-02-06 02:33:38 +03:00
|
|
|
llvm::FoldingSetNodeID ID;
|
Re-applies the patch first applied way back in r106099, with
accompanying fixes to make it work today.
The core of this patch is to provide a link from a TemplateTypeParmType
back to the TemplateTypeParmDecl node which declared it. This in turn
provides much more precise information about the type, where it came
from, and how it functions for AST consumers.
To make the patch work almost a year after its first attempt, it needed
serialization support, and it now retains the old getName() interface.
Finally, it requires us to not attempt to instantiate the type in an
unsupported friend decl -- specifically those coming from template
friend decls but which refer to a specific type through a dependent
name.
A cleaner representation of the last item would be to build
FriendTemplateDecl nodes for these, storing their template parameters
etc, and to perform proper instantation of them like any other template
declaration. They can still be flagged as unsupported for the purpose of
access checking, etc.
This passed an asserts-enabled bootstrap for me, and the reduced test
case mentioned in the original review thread no longer causes issues,
likely fixed at somewhere amidst the 24k revisions that have elapsed.
git-svn-id: https://llvm.org/svn/llvm-project/cfe/trunk@130628 91177308-0d34-0410-b5e6-96231b3b80d8
2011-05-01 04:51:33 +04:00
|
|
|
TemplateTypeParmType::Profile(ID, Depth, Index, ParameterPack, TTPDecl);
|
2009-02-06 02:33:38 +03:00
|
|
|
void *InsertPos = 0;
|
2009-09-09 19:08:12 +04:00
|
|
|
TemplateTypeParmType *TypeParm
|
2009-02-06 02:33:38 +03:00
|
|
|
= TemplateTypeParmTypes.FindNodeOrInsertPos(ID, InsertPos);
|
|
|
|
|
|
|
|
if (TypeParm)
|
|
|
|
return QualType(TypeParm, 0);
|
2009-09-09 19:08:12 +04:00
|
|
|
|
Re-applies the patch first applied way back in r106099, with
accompanying fixes to make it work today.
The core of this patch is to provide a link from a TemplateTypeParmType
back to the TemplateTypeParmDecl node which declared it. This in turn
provides much more precise information about the type, where it came
from, and how it functions for AST consumers.
To make the patch work almost a year after its first attempt, it needed
serialization support, and it now retains the old getName() interface.
Finally, it requires us to not attempt to instantiate the type in an
unsupported friend decl -- specifically those coming from template
friend decls but which refer to a specific type through a dependent
name.
A cleaner representation of the last item would be to build
FriendTemplateDecl nodes for these, storing their template parameters
etc, and to perform proper instantation of them like any other template
declaration. They can still be flagged as unsupported for the purpose of
access checking, etc.
This passed an asserts-enabled bootstrap for me, and the reduced test
case mentioned in the original review thread no longer causes issues,
likely fixed at somewhere amidst the 24k revisions that have elapsed.
git-svn-id: https://llvm.org/svn/llvm-project/cfe/trunk@130628 91177308-0d34-0410-b5e6-96231b3b80d8
2011-05-01 04:51:33 +04:00
|
|
|
if (TTPDecl) {
|
2009-06-16 04:30:48 +04:00
|
|
|
QualType Canon = getTemplateTypeParmType(Depth, Index, ParameterPack);
|
Re-applies the patch first applied way back in r106099, with
accompanying fixes to make it work today.
The core of this patch is to provide a link from a TemplateTypeParmType
back to the TemplateTypeParmDecl node which declared it. This in turn
provides much more precise information about the type, where it came
from, and how it functions for AST consumers.
To make the patch work almost a year after its first attempt, it needed
serialization support, and it now retains the old getName() interface.
Finally, it requires us to not attempt to instantiate the type in an
unsupported friend decl -- specifically those coming from template
friend decls but which refer to a specific type through a dependent
name.
A cleaner representation of the last item would be to build
FriendTemplateDecl nodes for these, storing their template parameters
etc, and to perform proper instantation of them like any other template
declaration. They can still be flagged as unsupported for the purpose of
access checking, etc.
This passed an asserts-enabled bootstrap for me, and the reduced test
case mentioned in the original review thread no longer causes issues,
likely fixed at somewhere amidst the 24k revisions that have elapsed.
git-svn-id: https://llvm.org/svn/llvm-project/cfe/trunk@130628 91177308-0d34-0410-b5e6-96231b3b80d8
2011-05-01 04:51:33 +04:00
|
|
|
TypeParm = new (*this, TypeAlignment) TemplateTypeParmType(TTPDecl, Canon);
|
Fix an obscure crash found in the Boost.MPL test suite, along with a
ton of potential crashes of the same kind. The fundamental problem is
that type creation was following a dangerous pattern when using its
FoldingSets:
1) Use FindNodeOrInsertPos to see if the type is available
2) If not, and we aren't looking at a canonical type, build the
canonical type
3) Build and insert the new node into the FoldingSet
The problem here is that building the canonical type can, in very rare
circumstances, force the hash table inside the FoldingSet to
reallocate. That invalidates the insertion position we computed in
step 1, and in step 3 we end up inserting the new node into the wrong
place. BOOM!
I've audited all of ASTContext, fixing this problem everywhere I found
it. The vast majority of wrong code was C++-specific (and *ahem*
written by me), so I also audited other major folding sets in the C++
code (e.g., template specializations), but found no other instances of
this problem.
git-svn-id: https://llvm.org/svn/llvm-project/cfe/trunk@95315 91177308-0d34-0410-b5e6-96231b3b80d8
2010-02-04 21:10:26 +03:00
|
|
|
|
|
|
|
TemplateTypeParmType *TypeCheck
|
|
|
|
= TemplateTypeParmTypes.FindNodeOrInsertPos(ID, InsertPos);
|
|
|
|
assert(!TypeCheck && "Template type parameter canonical type broken");
|
|
|
|
(void)TypeCheck;
|
2009-06-16 04:30:48 +04:00
|
|
|
} else
|
2009-09-25 03:30:46 +04:00
|
|
|
TypeParm = new (*this, TypeAlignment)
|
|
|
|
TemplateTypeParmType(Depth, Index, ParameterPack);
|
2009-02-06 02:33:38 +03:00
|
|
|
|
|
|
|
Types.push_back(TypeParm);
|
|
|
|
TemplateTypeParmTypes.InsertNode(TypeParm, InsertPos);
|
|
|
|
|
|
|
|
return QualType(TypeParm, 0);
|
|
|
|
}
|
|
|
|
|
2010-03-10 06:28:59 +03:00
|
|
|
TypeSourceInfo *
|
|
|
|
ASTContext::getTemplateSpecializationTypeInfo(TemplateName Name,
|
|
|
|
SourceLocation NameLoc,
|
|
|
|
const TemplateArgumentListInfo &Args,
|
2011-05-06 01:57:07 +04:00
|
|
|
QualType Underlying) const {
|
2011-02-28 21:50:33 +03:00
|
|
|
assert(!Name.getAsDependentTemplateName() &&
|
|
|
|
"No dependent template names here!");
|
2011-05-06 01:57:07 +04:00
|
|
|
QualType TST = getTemplateSpecializationType(Name, Args, Underlying);
|
2010-03-10 06:28:59 +03:00
|
|
|
|
|
|
|
TypeSourceInfo *DI = CreateTypeSourceInfo(TST);
|
|
|
|
TemplateSpecializationTypeLoc TL
|
|
|
|
= cast<TemplateSpecializationTypeLoc>(DI->getTypeLoc());
|
2012-02-06 18:41:24 +04:00
|
|
|
TL.setTemplateKeywordLoc(SourceLocation());
|
2010-03-10 06:28:59 +03:00
|
|
|
TL.setTemplateNameLoc(NameLoc);
|
|
|
|
TL.setLAngleLoc(Args.getLAngleLoc());
|
|
|
|
TL.setRAngleLoc(Args.getRAngleLoc());
|
|
|
|
for (unsigned i = 0, e = TL.getNumArgs(); i != e; ++i)
|
|
|
|
TL.setArgLocInfo(i, Args[i].getLocInfo());
|
|
|
|
return DI;
|
|
|
|
}
|
|
|
|
|
2009-10-29 11:12:44 +03:00
|
|
|
QualType
|
|
|
|
ASTContext::getTemplateSpecializationType(TemplateName Template,
|
2009-11-23 04:53:49 +03:00
|
|
|
const TemplateArgumentListInfo &Args,
|
2011-05-06 01:57:07 +04:00
|
|
|
QualType Underlying) const {
|
2011-02-28 21:50:33 +03:00
|
|
|
assert(!Template.getAsDependentTemplateName() &&
|
|
|
|
"No dependent template names here!");
|
|
|
|
|
2009-11-23 04:53:49 +03:00
|
|
|
unsigned NumArgs = Args.size();
|
|
|
|
|
2011-07-23 14:55:15 +04:00
|
|
|
SmallVector<TemplateArgument, 4> ArgVec;
|
2009-10-29 11:12:44 +03:00
|
|
|
ArgVec.reserve(NumArgs);
|
|
|
|
for (unsigned i = 0; i != NumArgs; ++i)
|
|
|
|
ArgVec.push_back(Args[i].getArgument());
|
|
|
|
|
2010-04-27 04:57:59 +04:00
|
|
|
return getTemplateSpecializationType(Template, ArgVec.data(), NumArgs,
|
2011-05-06 01:57:07 +04:00
|
|
|
Underlying);
|
2009-10-29 11:12:44 +03:00
|
|
|
}
|
|
|
|
|
2012-02-03 21:16:23 +04:00
|
|
|
#ifndef NDEBUG
|
|
|
|
static bool hasAnyPackExpansions(const TemplateArgument *Args,
|
|
|
|
unsigned NumArgs) {
|
|
|
|
for (unsigned I = 0; I != NumArgs; ++I)
|
|
|
|
if (Args[I].isPackExpansion())
|
|
|
|
return true;
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2009-09-09 19:08:12 +04:00
|
|
|
QualType
|
2009-03-31 02:58:21 +04:00
|
|
|
ASTContext::getTemplateSpecializationType(TemplateName Template,
|
|
|
|
const TemplateArgument *Args,
|
|
|
|
unsigned NumArgs,
|
2011-05-06 01:57:07 +04:00
|
|
|
QualType Underlying) const {
|
2011-02-28 21:50:33 +03:00
|
|
|
assert(!Template.getAsDependentTemplateName() &&
|
|
|
|
"No dependent template names here!");
|
2011-03-03 20:04:51 +03:00
|
|
|
// Look through qualified template names.
|
|
|
|
if (QualifiedTemplateName *QTN = Template.getAsQualifiedTemplateName())
|
|
|
|
Template = TemplateName(QTN->getTemplateDecl());
|
2011-02-28 21:50:33 +03:00
|
|
|
|
2012-02-03 21:16:23 +04:00
|
|
|
bool IsTypeAlias =
|
2011-05-06 01:57:07 +04:00
|
|
|
Template.getAsTemplateDecl() &&
|
|
|
|
isa<TypeAliasTemplateDecl>(Template.getAsTemplateDecl());
|
|
|
|
QualType CanonType;
|
|
|
|
if (!Underlying.isNull())
|
|
|
|
CanonType = getCanonicalType(Underlying);
|
|
|
|
else {
|
2012-02-03 21:16:23 +04:00
|
|
|
// We can get here with an alias template when the specialization contains
|
|
|
|
// a pack expansion that does not match up with a parameter pack.
|
|
|
|
assert((!IsTypeAlias || hasAnyPackExpansions(Args, NumArgs)) &&
|
|
|
|
"Caller must compute aliased type");
|
|
|
|
IsTypeAlias = false;
|
2011-05-06 01:57:07 +04:00
|
|
|
CanonType = getCanonicalTemplateSpecializationType(Template, Args,
|
|
|
|
NumArgs);
|
|
|
|
}
|
2009-02-27 01:19:44 +03:00
|
|
|
|
2009-07-29 03:00:59 +04:00
|
|
|
// Allocate the (non-canonical) template specialization type, but don't
|
|
|
|
// try to unique it: these types typically have location information that
|
|
|
|
// we don't unique and don't want to lose.
|
2011-05-06 01:57:07 +04:00
|
|
|
void *Mem = Allocate(sizeof(TemplateSpecializationType) +
|
|
|
|
sizeof(TemplateArgument) * NumArgs +
|
2012-02-03 21:16:23 +04:00
|
|
|
(IsTypeAlias? sizeof(QualType) : 0),
|
2009-09-25 03:30:46 +04:00
|
|
|
TypeAlignment);
|
2009-09-09 19:08:12 +04:00
|
|
|
TemplateSpecializationType *Spec
|
2012-02-03 21:16:23 +04:00
|
|
|
= new (Mem) TemplateSpecializationType(Template, Args, NumArgs, CanonType,
|
|
|
|
IsTypeAlias ? Underlying : QualType());
|
2009-09-09 19:08:12 +04:00
|
|
|
|
2009-02-09 21:46:07 +03:00
|
|
|
Types.push_back(Spec);
|
2009-09-09 19:08:12 +04:00
|
|
|
return QualType(Spec, 0);
|
2009-02-09 21:46:07 +03:00
|
|
|
}
|
|
|
|
|
2010-07-02 15:55:11 +04:00
|
|
|
QualType
|
|
|
|
ASTContext::getCanonicalTemplateSpecializationType(TemplateName Template,
|
|
|
|
const TemplateArgument *Args,
|
2011-01-12 12:06:06 +03:00
|
|
|
unsigned NumArgs) const {
|
2011-02-28 21:50:33 +03:00
|
|
|
assert(!Template.getAsDependentTemplateName() &&
|
|
|
|
"No dependent template names here!");
|
2011-05-06 01:57:07 +04:00
|
|
|
|
2011-03-03 20:04:51 +03:00
|
|
|
// Look through qualified template names.
|
|
|
|
if (QualifiedTemplateName *QTN = Template.getAsQualifiedTemplateName())
|
|
|
|
Template = TemplateName(QTN->getTemplateDecl());
|
2011-02-28 21:50:33 +03:00
|
|
|
|
2010-07-02 15:55:11 +04:00
|
|
|
// Build the canonical template specialization type.
|
|
|
|
TemplateName CanonTemplate = getCanonicalTemplateName(Template);
|
2011-07-23 14:55:15 +04:00
|
|
|
SmallVector<TemplateArgument, 4> CanonArgs;
|
2010-07-02 15:55:11 +04:00
|
|
|
CanonArgs.reserve(NumArgs);
|
|
|
|
for (unsigned I = 0; I != NumArgs; ++I)
|
|
|
|
CanonArgs.push_back(getCanonicalTemplateArgument(Args[I]));
|
|
|
|
|
|
|
|
// Determine whether this canonical template specialization type already
|
|
|
|
// exists.
|
|
|
|
llvm::FoldingSetNodeID ID;
|
|
|
|
TemplateSpecializationType::Profile(ID, CanonTemplate,
|
|
|
|
CanonArgs.data(), NumArgs, *this);
|
|
|
|
|
|
|
|
void *InsertPos = 0;
|
|
|
|
TemplateSpecializationType *Spec
|
|
|
|
= TemplateSpecializationTypes.FindNodeOrInsertPos(ID, InsertPos);
|
|
|
|
|
|
|
|
if (!Spec) {
|
|
|
|
// Allocate a new canonical template specialization type.
|
|
|
|
void *Mem = Allocate((sizeof(TemplateSpecializationType) +
|
|
|
|
sizeof(TemplateArgument) * NumArgs),
|
|
|
|
TypeAlignment);
|
|
|
|
Spec = new (Mem) TemplateSpecializationType(CanonTemplate,
|
|
|
|
CanonArgs.data(), NumArgs,
|
2011-05-06 01:57:07 +04:00
|
|
|
QualType(), QualType());
|
2010-07-02 15:55:11 +04:00
|
|
|
Types.push_back(Spec);
|
|
|
|
TemplateSpecializationTypes.InsertNode(Spec, InsertPos);
|
|
|
|
}
|
|
|
|
|
|
|
|
assert(Spec->isDependentType() &&
|
|
|
|
"Non-dependent template-id type must have a canonical type");
|
|
|
|
return QualType(Spec, 0);
|
|
|
|
}
|
|
|
|
|
2009-09-09 19:08:12 +04:00
|
|
|
QualType
|
2010-05-12 01:36:43 +04:00
|
|
|
ASTContext::getElaboratedType(ElaboratedTypeKeyword Keyword,
|
|
|
|
NestedNameSpecifier *NNS,
|
2011-01-12 12:06:06 +03:00
|
|
|
QualType NamedType) const {
|
Introduce a representation for types that we referred to via a
qualified name, e.g.,
foo::x
so that we retain the nested-name-specifier as written in the source
code and can reproduce that qualified name when printing the types
back (e.g., in diagnostics). This is PR3493, which won't be complete
until finished the other tasks mentioned near the end of this commit.
The parser's representation of nested-name-specifiers, CXXScopeSpec,
is now a bit fatter, because it needs to contain the scopes that
precede each '::' and keep track of whether the global scoping
operator '::' was at the beginning. For example, we need to keep track
of the leading '::', 'foo', and 'bar' in
::foo::bar::x
The Action's CXXScopeTy * is no longer a DeclContext *. It's now the
opaque version of the new NestedNameSpecifier, which contains a single
component of a nested-name-specifier (either a DeclContext * or a Type
*, bitmangled).
The new sugar type QualifiedNameType composes a sequence of
NestedNameSpecifiers with a representation of the type we're actually
referring to. At present, we only build QualifiedNameType nodes within
Sema::getTypeName. This will be extended to other type-constructing
actions (e.g., ActOnClassTemplateId).
Also on the way: QualifiedDeclRefExprs will also store a sequence of
NestedNameSpecifiers, so that we can print out the property
nested-name-specifier. I expect to also use this for handling
dependent names like Fibonacci<I - 1>::value.
git-svn-id: https://llvm.org/svn/llvm-project/cfe/trunk@67265 91177308-0d34-0410-b5e6-96231b3b80d8
2009-03-19 03:18:19 +03:00
|
|
|
llvm::FoldingSetNodeID ID;
|
2010-05-12 01:36:43 +04:00
|
|
|
ElaboratedType::Profile(ID, Keyword, NNS, NamedType);
|
Introduce a representation for types that we referred to via a
qualified name, e.g.,
foo::x
so that we retain the nested-name-specifier as written in the source
code and can reproduce that qualified name when printing the types
back (e.g., in diagnostics). This is PR3493, which won't be complete
until finished the other tasks mentioned near the end of this commit.
The parser's representation of nested-name-specifiers, CXXScopeSpec,
is now a bit fatter, because it needs to contain the scopes that
precede each '::' and keep track of whether the global scoping
operator '::' was at the beginning. For example, we need to keep track
of the leading '::', 'foo', and 'bar' in
::foo::bar::x
The Action's CXXScopeTy * is no longer a DeclContext *. It's now the
opaque version of the new NestedNameSpecifier, which contains a single
component of a nested-name-specifier (either a DeclContext * or a Type
*, bitmangled).
The new sugar type QualifiedNameType composes a sequence of
NestedNameSpecifiers with a representation of the type we're actually
referring to. At present, we only build QualifiedNameType nodes within
Sema::getTypeName. This will be extended to other type-constructing
actions (e.g., ActOnClassTemplateId).
Also on the way: QualifiedDeclRefExprs will also store a sequence of
NestedNameSpecifiers, so that we can print out the property
nested-name-specifier. I expect to also use this for handling
dependent names like Fibonacci<I - 1>::value.
git-svn-id: https://llvm.org/svn/llvm-project/cfe/trunk@67265 91177308-0d34-0410-b5e6-96231b3b80d8
2009-03-19 03:18:19 +03:00
|
|
|
|
|
|
|
void *InsertPos = 0;
|
2010-05-12 01:36:43 +04:00
|
|
|
ElaboratedType *T = ElaboratedTypes.FindNodeOrInsertPos(ID, InsertPos);
|
Introduce a representation for types that we referred to via a
qualified name, e.g.,
foo::x
so that we retain the nested-name-specifier as written in the source
code and can reproduce that qualified name when printing the types
back (e.g., in diagnostics). This is PR3493, which won't be complete
until finished the other tasks mentioned near the end of this commit.
The parser's representation of nested-name-specifiers, CXXScopeSpec,
is now a bit fatter, because it needs to contain the scopes that
precede each '::' and keep track of whether the global scoping
operator '::' was at the beginning. For example, we need to keep track
of the leading '::', 'foo', and 'bar' in
::foo::bar::x
The Action's CXXScopeTy * is no longer a DeclContext *. It's now the
opaque version of the new NestedNameSpecifier, which contains a single
component of a nested-name-specifier (either a DeclContext * or a Type
*, bitmangled).
The new sugar type QualifiedNameType composes a sequence of
NestedNameSpecifiers with a representation of the type we're actually
referring to. At present, we only build QualifiedNameType nodes within
Sema::getTypeName. This will be extended to other type-constructing
actions (e.g., ActOnClassTemplateId).
Also on the way: QualifiedDeclRefExprs will also store a sequence of
NestedNameSpecifiers, so that we can print out the property
nested-name-specifier. I expect to also use this for handling
dependent names like Fibonacci<I - 1>::value.
git-svn-id: https://llvm.org/svn/llvm-project/cfe/trunk@67265 91177308-0d34-0410-b5e6-96231b3b80d8
2009-03-19 03:18:19 +03:00
|
|
|
if (T)
|
|
|
|
return QualType(T, 0);
|
|
|
|
|
Fix an obscure crash found in the Boost.MPL test suite, along with a
ton of potential crashes of the same kind. The fundamental problem is
that type creation was following a dangerous pattern when using its
FoldingSets:
1) Use FindNodeOrInsertPos to see if the type is available
2) If not, and we aren't looking at a canonical type, build the
canonical type
3) Build and insert the new node into the FoldingSet
The problem here is that building the canonical type can, in very rare
circumstances, force the hash table inside the FoldingSet to
reallocate. That invalidates the insertion position we computed in
step 1, and in step 3 we end up inserting the new node into the wrong
place. BOOM!
I've audited all of ASTContext, fixing this problem everywhere I found
it. The vast majority of wrong code was C++-specific (and *ahem*
written by me), so I also audited other major folding sets in the C++
code (e.g., template specializations), but found no other instances of
this problem.
git-svn-id: https://llvm.org/svn/llvm-project/cfe/trunk@95315 91177308-0d34-0410-b5e6-96231b3b80d8
2010-02-04 21:10:26 +03:00
|
|
|
QualType Canon = NamedType;
|
|
|
|
if (!Canon.isCanonical()) {
|
|
|
|
Canon = getCanonicalType(NamedType);
|
2010-05-12 01:36:43 +04:00
|
|
|
ElaboratedType *CheckT = ElaboratedTypes.FindNodeOrInsertPos(ID, InsertPos);
|
|
|
|
assert(!CheckT && "Elaborated canonical type broken");
|
Fix an obscure crash found in the Boost.MPL test suite, along with a
ton of potential crashes of the same kind. The fundamental problem is
that type creation was following a dangerous pattern when using its
FoldingSets:
1) Use FindNodeOrInsertPos to see if the type is available
2) If not, and we aren't looking at a canonical type, build the
canonical type
3) Build and insert the new node into the FoldingSet
The problem here is that building the canonical type can, in very rare
circumstances, force the hash table inside the FoldingSet to
reallocate. That invalidates the insertion position we computed in
step 1, and in step 3 we end up inserting the new node into the wrong
place. BOOM!
I've audited all of ASTContext, fixing this problem everywhere I found
it. The vast majority of wrong code was C++-specific (and *ahem*
written by me), so I also audited other major folding sets in the C++
code (e.g., template specializations), but found no other instances of
this problem.
git-svn-id: https://llvm.org/svn/llvm-project/cfe/trunk@95315 91177308-0d34-0410-b5e6-96231b3b80d8
2010-02-04 21:10:26 +03:00
|
|
|
(void)CheckT;
|
|
|
|
}
|
|
|
|
|
2010-05-12 01:36:43 +04:00
|
|
|
T = new (*this) ElaboratedType(Keyword, NNS, NamedType, Canon);
|
Introduce a representation for types that we referred to via a
qualified name, e.g.,
foo::x
so that we retain the nested-name-specifier as written in the source
code and can reproduce that qualified name when printing the types
back (e.g., in diagnostics). This is PR3493, which won't be complete
until finished the other tasks mentioned near the end of this commit.
The parser's representation of nested-name-specifiers, CXXScopeSpec,
is now a bit fatter, because it needs to contain the scopes that
precede each '::' and keep track of whether the global scoping
operator '::' was at the beginning. For example, we need to keep track
of the leading '::', 'foo', and 'bar' in
::foo::bar::x
The Action's CXXScopeTy * is no longer a DeclContext *. It's now the
opaque version of the new NestedNameSpecifier, which contains a single
component of a nested-name-specifier (either a DeclContext * or a Type
*, bitmangled).
The new sugar type QualifiedNameType composes a sequence of
NestedNameSpecifiers with a representation of the type we're actually
referring to. At present, we only build QualifiedNameType nodes within
Sema::getTypeName. This will be extended to other type-constructing
actions (e.g., ActOnClassTemplateId).
Also on the way: QualifiedDeclRefExprs will also store a sequence of
NestedNameSpecifiers, so that we can print out the property
nested-name-specifier. I expect to also use this for handling
dependent names like Fibonacci<I - 1>::value.
git-svn-id: https://llvm.org/svn/llvm-project/cfe/trunk@67265 91177308-0d34-0410-b5e6-96231b3b80d8
2009-03-19 03:18:19 +03:00
|
|
|
Types.push_back(T);
|
2010-05-12 01:36:43 +04:00
|
|
|
ElaboratedTypes.InsertNode(T, InsertPos);
|
Introduce a representation for types that we referred to via a
qualified name, e.g.,
foo::x
so that we retain the nested-name-specifier as written in the source
code and can reproduce that qualified name when printing the types
back (e.g., in diagnostics). This is PR3493, which won't be complete
until finished the other tasks mentioned near the end of this commit.
The parser's representation of nested-name-specifiers, CXXScopeSpec,
is now a bit fatter, because it needs to contain the scopes that
precede each '::' and keep track of whether the global scoping
operator '::' was at the beginning. For example, we need to keep track
of the leading '::', 'foo', and 'bar' in
::foo::bar::x
The Action's CXXScopeTy * is no longer a DeclContext *. It's now the
opaque version of the new NestedNameSpecifier, which contains a single
component of a nested-name-specifier (either a DeclContext * or a Type
*, bitmangled).
The new sugar type QualifiedNameType composes a sequence of
NestedNameSpecifiers with a representation of the type we're actually
referring to. At present, we only build QualifiedNameType nodes within
Sema::getTypeName. This will be extended to other type-constructing
actions (e.g., ActOnClassTemplateId).
Also on the way: QualifiedDeclRefExprs will also store a sequence of
NestedNameSpecifiers, so that we can print out the property
nested-name-specifier. I expect to also use this for handling
dependent names like Fibonacci<I - 1>::value.
git-svn-id: https://llvm.org/svn/llvm-project/cfe/trunk@67265 91177308-0d34-0410-b5e6-96231b3b80d8
2009-03-19 03:18:19 +03:00
|
|
|
return QualType(T, 0);
|
|
|
|
}
|
|
|
|
|
2010-12-10 19:29:40 +03:00
|
|
|
QualType
|
2011-01-12 12:06:06 +03:00
|
|
|
ASTContext::getParenType(QualType InnerType) const {
|
2010-12-10 19:29:40 +03:00
|
|
|
llvm::FoldingSetNodeID ID;
|
|
|
|
ParenType::Profile(ID, InnerType);
|
|
|
|
|
|
|
|
void *InsertPos = 0;
|
|
|
|
ParenType *T = ParenTypes.FindNodeOrInsertPos(ID, InsertPos);
|
|
|
|
if (T)
|
|
|
|
return QualType(T, 0);
|
|
|
|
|
|
|
|
QualType Canon = InnerType;
|
|
|
|
if (!Canon.isCanonical()) {
|
|
|
|
Canon = getCanonicalType(InnerType);
|
|
|
|
ParenType *CheckT = ParenTypes.FindNodeOrInsertPos(ID, InsertPos);
|
|
|
|
assert(!CheckT && "Paren canonical type broken");
|
|
|
|
(void)CheckT;
|
|
|
|
}
|
|
|
|
|
|
|
|
T = new (*this) ParenType(InnerType, Canon);
|
|
|
|
Types.push_back(T);
|
|
|
|
ParenTypes.InsertNode(T, InsertPos);
|
|
|
|
return QualType(T, 0);
|
|
|
|
}
|
|
|
|
|
2010-04-01 00:19:30 +04:00
|
|
|
QualType ASTContext::getDependentNameType(ElaboratedTypeKeyword Keyword,
|
|
|
|
NestedNameSpecifier *NNS,
|
|
|
|
const IdentifierInfo *Name,
|
2011-01-12 12:06:06 +03:00
|
|
|
QualType Canon) const {
|
2009-03-28 02:10:48 +03:00
|
|
|
assert(NNS->isDependent() && "nested-name-specifier must be dependent");
|
|
|
|
|
|
|
|
if (Canon.isNull()) {
|
|
|
|
NestedNameSpecifier *CanonNNS = getCanonicalNestedNameSpecifier(NNS);
|
2010-04-01 00:19:30 +04:00
|
|
|
ElaboratedTypeKeyword CanonKeyword = Keyword;
|
|
|
|
if (Keyword == ETK_None)
|
|
|
|
CanonKeyword = ETK_Typename;
|
|
|
|
|
|
|
|
if (CanonNNS != NNS || CanonKeyword != Keyword)
|
|
|
|
Canon = getDependentNameType(CanonKeyword, CanonNNS, Name);
|
2009-03-28 02:10:48 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
llvm::FoldingSetNodeID ID;
|
2010-04-01 00:19:30 +04:00
|
|
|
DependentNameType::Profile(ID, Keyword, NNS, Name);
|
2009-03-28 02:10:48 +03:00
|
|
|
|
|
|
|
void *InsertPos = 0;
|
2010-03-31 21:34:00 +04:00
|
|
|
DependentNameType *T
|
|
|
|
= DependentNameTypes.FindNodeOrInsertPos(ID, InsertPos);
|
2009-03-28 02:10:48 +03:00
|
|
|
if (T)
|
|
|
|
return QualType(T, 0);
|
|
|
|
|
2010-04-01 00:19:30 +04:00
|
|
|
T = new (*this) DependentNameType(Keyword, NNS, Name, Canon);
|
2009-03-28 02:10:48 +03:00
|
|
|
Types.push_back(T);
|
2010-03-31 21:34:00 +04:00
|
|
|
DependentNameTypes.InsertNode(T, InsertPos);
|
2009-09-09 19:08:12 +04:00
|
|
|
return QualType(T, 0);
|
2009-03-28 02:10:48 +03:00
|
|
|
}
|
|
|
|
|
2009-09-09 19:08:12 +04:00
|
|
|
QualType
|
2010-06-11 04:33:02 +04:00
|
|
|
ASTContext::getDependentTemplateSpecializationType(
|
|
|
|
ElaboratedTypeKeyword Keyword,
|
2010-04-01 00:19:30 +04:00
|
|
|
NestedNameSpecifier *NNS,
|
2010-06-11 04:33:02 +04:00
|
|
|
const IdentifierInfo *Name,
|
2011-01-12 12:06:06 +03:00
|
|
|
const TemplateArgumentListInfo &Args) const {
|
2010-06-11 04:33:02 +04:00
|
|
|
// TODO: avoid this copy
|
2011-07-23 14:55:15 +04:00
|
|
|
SmallVector<TemplateArgument, 16> ArgCopy;
|
2010-06-11 04:33:02 +04:00
|
|
|
for (unsigned I = 0, E = Args.size(); I != E; ++I)
|
|
|
|
ArgCopy.push_back(Args[I].getArgument());
|
|
|
|
return getDependentTemplateSpecializationType(Keyword, NNS, Name,
|
|
|
|
ArgCopy.size(),
|
|
|
|
ArgCopy.data());
|
|
|
|
}
|
|
|
|
|
|
|
|
QualType
|
|
|
|
ASTContext::getDependentTemplateSpecializationType(
|
|
|
|
ElaboratedTypeKeyword Keyword,
|
|
|
|
NestedNameSpecifier *NNS,
|
|
|
|
const IdentifierInfo *Name,
|
|
|
|
unsigned NumArgs,
|
2011-01-12 12:06:06 +03:00
|
|
|
const TemplateArgument *Args) const {
|
2011-02-28 03:04:36 +03:00
|
|
|
assert((!NNS || NNS->isDependent()) &&
|
|
|
|
"nested-name-specifier must be dependent");
|
2009-04-01 04:28:59 +04:00
|
|
|
|
Fix an obscure crash found in the Boost.MPL test suite, along with a
ton of potential crashes of the same kind. The fundamental problem is
that type creation was following a dangerous pattern when using its
FoldingSets:
1) Use FindNodeOrInsertPos to see if the type is available
2) If not, and we aren't looking at a canonical type, build the
canonical type
3) Build and insert the new node into the FoldingSet
The problem here is that building the canonical type can, in very rare
circumstances, force the hash table inside the FoldingSet to
reallocate. That invalidates the insertion position we computed in
step 1, and in step 3 we end up inserting the new node into the wrong
place. BOOM!
I've audited all of ASTContext, fixing this problem everywhere I found
it. The vast majority of wrong code was C++-specific (and *ahem*
written by me), so I also audited other major folding sets in the C++
code (e.g., template specializations), but found no other instances of
this problem.
git-svn-id: https://llvm.org/svn/llvm-project/cfe/trunk@95315 91177308-0d34-0410-b5e6-96231b3b80d8
2010-02-04 21:10:26 +03:00
|
|
|
llvm::FoldingSetNodeID ID;
|
2010-06-11 04:33:02 +04:00
|
|
|
DependentTemplateSpecializationType::Profile(ID, *this, Keyword, NNS,
|
|
|
|
Name, NumArgs, Args);
|
Fix an obscure crash found in the Boost.MPL test suite, along with a
ton of potential crashes of the same kind. The fundamental problem is
that type creation was following a dangerous pattern when using its
FoldingSets:
1) Use FindNodeOrInsertPos to see if the type is available
2) If not, and we aren't looking at a canonical type, build the
canonical type
3) Build and insert the new node into the FoldingSet
The problem here is that building the canonical type can, in very rare
circumstances, force the hash table inside the FoldingSet to
reallocate. That invalidates the insertion position we computed in
step 1, and in step 3 we end up inserting the new node into the wrong
place. BOOM!
I've audited all of ASTContext, fixing this problem everywhere I found
it. The vast majority of wrong code was C++-specific (and *ahem*
written by me), so I also audited other major folding sets in the C++
code (e.g., template specializations), but found no other instances of
this problem.
git-svn-id: https://llvm.org/svn/llvm-project/cfe/trunk@95315 91177308-0d34-0410-b5e6-96231b3b80d8
2010-02-04 21:10:26 +03:00
|
|
|
|
|
|
|
void *InsertPos = 0;
|
2010-06-11 04:33:02 +04:00
|
|
|
DependentTemplateSpecializationType *T
|
|
|
|
= DependentTemplateSpecializationTypes.FindNodeOrInsertPos(ID, InsertPos);
|
Fix an obscure crash found in the Boost.MPL test suite, along with a
ton of potential crashes of the same kind. The fundamental problem is
that type creation was following a dangerous pattern when using its
FoldingSets:
1) Use FindNodeOrInsertPos to see if the type is available
2) If not, and we aren't looking at a canonical type, build the
canonical type
3) Build and insert the new node into the FoldingSet
The problem here is that building the canonical type can, in very rare
circumstances, force the hash table inside the FoldingSet to
reallocate. That invalidates the insertion position we computed in
step 1, and in step 3 we end up inserting the new node into the wrong
place. BOOM!
I've audited all of ASTContext, fixing this problem everywhere I found
it. The vast majority of wrong code was C++-specific (and *ahem*
written by me), so I also audited other major folding sets in the C++
code (e.g., template specializations), but found no other instances of
this problem.
git-svn-id: https://llvm.org/svn/llvm-project/cfe/trunk@95315 91177308-0d34-0410-b5e6-96231b3b80d8
2010-02-04 21:10:26 +03:00
|
|
|
if (T)
|
|
|
|
return QualType(T, 0);
|
|
|
|
|
2010-06-11 04:33:02 +04:00
|
|
|
NestedNameSpecifier *CanonNNS = getCanonicalNestedNameSpecifier(NNS);
|
|
|
|
|
|
|
|
ElaboratedTypeKeyword CanonKeyword = Keyword;
|
|
|
|
if (Keyword == ETK_None) CanonKeyword = ETK_Typename;
|
2009-04-01 04:28:59 +04:00
|
|
|
|
2010-06-11 04:33:02 +04:00
|
|
|
bool AnyNonCanonArgs = false;
|
2011-07-23 14:55:15 +04:00
|
|
|
SmallVector<TemplateArgument, 16> CanonArgs(NumArgs);
|
2010-06-11 04:33:02 +04:00
|
|
|
for (unsigned I = 0; I != NumArgs; ++I) {
|
|
|
|
CanonArgs[I] = getCanonicalTemplateArgument(Args[I]);
|
|
|
|
if (!CanonArgs[I].structurallyEquals(Args[I]))
|
|
|
|
AnyNonCanonArgs = true;
|
Fix an obscure crash found in the Boost.MPL test suite, along with a
ton of potential crashes of the same kind. The fundamental problem is
that type creation was following a dangerous pattern when using its
FoldingSets:
1) Use FindNodeOrInsertPos to see if the type is available
2) If not, and we aren't looking at a canonical type, build the
canonical type
3) Build and insert the new node into the FoldingSet
The problem here is that building the canonical type can, in very rare
circumstances, force the hash table inside the FoldingSet to
reallocate. That invalidates the insertion position we computed in
step 1, and in step 3 we end up inserting the new node into the wrong
place. BOOM!
I've audited all of ASTContext, fixing this problem everywhere I found
it. The vast majority of wrong code was C++-specific (and *ahem*
written by me), so I also audited other major folding sets in the C++
code (e.g., template specializations), but found no other instances of
this problem.
git-svn-id: https://llvm.org/svn/llvm-project/cfe/trunk@95315 91177308-0d34-0410-b5e6-96231b3b80d8
2010-02-04 21:10:26 +03:00
|
|
|
}
|
2009-04-01 04:28:59 +04:00
|
|
|
|
2010-06-11 04:33:02 +04:00
|
|
|
QualType Canon;
|
|
|
|
if (AnyNonCanonArgs || CanonNNS != NNS || CanonKeyword != Keyword) {
|
|
|
|
Canon = getDependentTemplateSpecializationType(CanonKeyword, CanonNNS,
|
|
|
|
Name, NumArgs,
|
|
|
|
CanonArgs.data());
|
|
|
|
|
|
|
|
// Find the insert position again.
|
|
|
|
DependentTemplateSpecializationTypes.FindNodeOrInsertPos(ID, InsertPos);
|
|
|
|
}
|
|
|
|
|
|
|
|
void *Mem = Allocate((sizeof(DependentTemplateSpecializationType) +
|
|
|
|
sizeof(TemplateArgument) * NumArgs),
|
|
|
|
TypeAlignment);
|
2010-06-11 15:07:21 +04:00
|
|
|
T = new (Mem) DependentTemplateSpecializationType(Keyword, NNS,
|
2010-06-11 04:33:02 +04:00
|
|
|
Name, NumArgs, Args, Canon);
|
2009-04-01 04:28:59 +04:00
|
|
|
Types.push_back(T);
|
2010-06-11 04:33:02 +04:00
|
|
|
DependentTemplateSpecializationTypes.InsertNode(T, InsertPos);
|
2009-09-09 19:08:12 +04:00
|
|
|
return QualType(T, 0);
|
2009-04-01 04:28:59 +04:00
|
|
|
}
|
|
|
|
|
2011-01-14 20:04:44 +03:00
|
|
|
QualType ASTContext::getPackExpansionType(QualType Pattern,
|
|
|
|
llvm::Optional<unsigned> NumExpansions) {
|
2010-12-20 05:24:11 +03:00
|
|
|
llvm::FoldingSetNodeID ID;
|
2011-01-14 20:04:44 +03:00
|
|
|
PackExpansionType::Profile(ID, Pattern, NumExpansions);
|
2010-12-20 05:24:11 +03:00
|
|
|
|
|
|
|
assert(Pattern->containsUnexpandedParameterPack() &&
|
|
|
|
"Pack expansions must expand one or more parameter packs");
|
|
|
|
void *InsertPos = 0;
|
|
|
|
PackExpansionType *T
|
|
|
|
= PackExpansionTypes.FindNodeOrInsertPos(ID, InsertPos);
|
|
|
|
if (T)
|
|
|
|
return QualType(T, 0);
|
|
|
|
|
|
|
|
QualType Canon;
|
|
|
|
if (!Pattern.isCanonical()) {
|
2011-01-14 20:04:44 +03:00
|
|
|
Canon = getPackExpansionType(getCanonicalType(Pattern), NumExpansions);
|
2010-12-20 05:24:11 +03:00
|
|
|
|
|
|
|
// Find the insert position again.
|
|
|
|
PackExpansionTypes.FindNodeOrInsertPos(ID, InsertPos);
|
|
|
|
}
|
|
|
|
|
2011-01-14 20:04:44 +03:00
|
|
|
T = new (*this) PackExpansionType(Pattern, Canon, NumExpansions);
|
2010-12-20 05:24:11 +03:00
|
|
|
Types.push_back(T);
|
|
|
|
PackExpansionTypes.InsertNode(T, InsertPos);
|
|
|
|
return QualType(T, 0);
|
|
|
|
}
|
|
|
|
|
2008-04-07 08:56:42 +04:00
|
|
|
/// CmpProtocolNames - Comparison predicate for sorting protocols
|
|
|
|
/// alphabetically.
|
|
|
|
static bool CmpProtocolNames(const ObjCProtocolDecl *LHS,
|
|
|
|
const ObjCProtocolDecl *RHS) {
|
2008-11-17 17:58:09 +03:00
|
|
|
return LHS->getDeclName() < RHS->getDeclName();
|
2008-04-07 08:56:42 +04:00
|
|
|
}
|
|
|
|
|
2010-05-15 15:32:37 +04:00
|
|
|
static bool areSortedAndUniqued(ObjCProtocolDecl * const *Protocols,
|
2009-10-23 02:37:11 +04:00
|
|
|
unsigned NumProtocols) {
|
|
|
|
if (NumProtocols == 0) return true;
|
|
|
|
|
2012-01-02 06:00:30 +04:00
|
|
|
if (Protocols[0]->getCanonicalDecl() != Protocols[0])
|
|
|
|
return false;
|
|
|
|
|
2009-10-23 02:37:11 +04:00
|
|
|
for (unsigned i = 1; i != NumProtocols; ++i)
|
2012-01-02 06:00:30 +04:00
|
|
|
if (!CmpProtocolNames(Protocols[i-1], Protocols[i]) ||
|
|
|
|
Protocols[i]->getCanonicalDecl() != Protocols[i])
|
2009-10-23 02:37:11 +04:00
|
|
|
return false;
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void SortAndUniqueProtocols(ObjCProtocolDecl **Protocols,
|
2008-04-07 08:56:42 +04:00
|
|
|
unsigned &NumProtocols) {
|
|
|
|
ObjCProtocolDecl **ProtocolsEnd = Protocols+NumProtocols;
|
2009-09-09 19:08:12 +04:00
|
|
|
|
2008-04-07 08:56:42 +04:00
|
|
|
// Sort protocols, keyed by name.
|
|
|
|
std::sort(Protocols, Protocols+NumProtocols, CmpProtocolNames);
|
|
|
|
|
2012-01-02 06:00:30 +04:00
|
|
|
// Canonicalize.
|
|
|
|
for (unsigned I = 0, N = NumProtocols; I != N; ++I)
|
|
|
|
Protocols[I] = Protocols[I]->getCanonicalDecl();
|
|
|
|
|
2008-04-07 08:56:42 +04:00
|
|
|
// Remove duplicates.
|
|
|
|
ProtocolsEnd = std::unique(Protocols, ProtocolsEnd);
|
|
|
|
NumProtocols = ProtocolsEnd-Protocols;
|
|
|
|
}
|
|
|
|
|
2010-05-15 15:32:37 +04:00
|
|
|
QualType ASTContext::getObjCObjectType(QualType BaseType,
|
|
|
|
ObjCProtocolDecl * const *Protocols,
|
2011-01-12 12:06:06 +03:00
|
|
|
unsigned NumProtocols) const {
|
2010-05-15 15:32:37 +04:00
|
|
|
// If the base type is an interface and there aren't any protocols
|
|
|
|
// to add, then the interface type will do just fine.
|
|
|
|
if (!NumProtocols && isa<ObjCInterfaceType>(BaseType))
|
|
|
|
return BaseType;
|
2009-06-18 02:40:22 +04:00
|
|
|
|
2010-05-15 15:32:37 +04:00
|
|
|
// Look in the folding set for an existing type.
|
|
|
|
llvm::FoldingSetNodeID ID;
|
|
|
|
ObjCObjectTypeImpl::Profile(ID, BaseType, Protocols, NumProtocols);
|
2009-06-18 02:40:22 +04:00
|
|
|
void *InsertPos = 0;
|
2010-05-15 15:32:37 +04:00
|
|
|
if (ObjCObjectType *QT = ObjCObjectTypes.FindNodeOrInsertPos(ID, InsertPos))
|
|
|
|
return QualType(QT, 0);
|
2009-06-18 02:40:22 +04:00
|
|
|
|
2010-05-15 15:32:37 +04:00
|
|
|
// Build the canonical type, which has the canonical base type and
|
|
|
|
// a sorted-and-uniqued list of protocols.
|
2009-10-23 02:37:11 +04:00
|
|
|
QualType Canonical;
|
2010-05-15 15:32:37 +04:00
|
|
|
bool ProtocolsSorted = areSortedAndUniqued(Protocols, NumProtocols);
|
|
|
|
if (!ProtocolsSorted || !BaseType.isCanonical()) {
|
|
|
|
if (!ProtocolsSorted) {
|
2011-07-23 14:55:15 +04:00
|
|
|
SmallVector<ObjCProtocolDecl*, 8> Sorted(Protocols,
|
2010-04-27 21:12:11 +04:00
|
|
|
Protocols + NumProtocols);
|
2009-10-23 02:37:11 +04:00
|
|
|
unsigned UniqueCount = NumProtocols;
|
|
|
|
|
|
|
|
SortAndUniqueProtocols(&Sorted[0], UniqueCount);
|
2010-05-15 15:32:37 +04:00
|
|
|
Canonical = getObjCObjectType(getCanonicalType(BaseType),
|
|
|
|
&Sorted[0], UniqueCount);
|
2009-10-23 02:37:11 +04:00
|
|
|
} else {
|
2010-05-15 15:32:37 +04:00
|
|
|
Canonical = getObjCObjectType(getCanonicalType(BaseType),
|
|
|
|
Protocols, NumProtocols);
|
2009-10-23 02:37:11 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
// Regenerate InsertPos.
|
2010-05-15 15:32:37 +04:00
|
|
|
ObjCObjectTypes.FindNodeOrInsertPos(ID, InsertPos);
|
2009-10-23 02:37:11 +04:00
|
|
|
}
|
|
|
|
|
2010-05-15 15:32:37 +04:00
|
|
|
unsigned Size = sizeof(ObjCObjectTypeImpl);
|
|
|
|
Size += NumProtocols * sizeof(ObjCProtocolDecl *);
|
2010-02-09 01:59:26 +03:00
|
|
|
void *Mem = Allocate(Size, TypeAlignment);
|
2010-05-15 15:32:37 +04:00
|
|
|
ObjCObjectTypeImpl *T =
|
|
|
|
new (Mem) ObjCObjectTypeImpl(Canonical, BaseType, Protocols, NumProtocols);
|
2009-09-09 19:08:12 +04:00
|
|
|
|
2010-05-15 15:32:37 +04:00
|
|
|
Types.push_back(T);
|
|
|
|
ObjCObjectTypes.InsertNode(T, InsertPos);
|
|
|
|
return QualType(T, 0);
|
2009-06-18 02:40:22 +04:00
|
|
|
}
|
2008-04-07 08:56:42 +04:00
|
|
|
|
2010-05-15 15:32:37 +04:00
|
|
|
/// getObjCObjectPointerType - Return a ObjCObjectPointerType type for
|
|
|
|
/// the given object type.
|
2011-01-12 12:06:06 +03:00
|
|
|
QualType ASTContext::getObjCObjectPointerType(QualType ObjectT) const {
|
2007-10-11 04:55:41 +04:00
|
|
|
llvm::FoldingSetNodeID ID;
|
2010-05-15 15:32:37 +04:00
|
|
|
ObjCObjectPointerType::Profile(ID, ObjectT);
|
2009-09-09 19:08:12 +04:00
|
|
|
|
2007-10-11 04:55:41 +04:00
|
|
|
void *InsertPos = 0;
|
2010-05-15 15:32:37 +04:00
|
|
|
if (ObjCObjectPointerType *QT =
|
|
|
|
ObjCObjectPointerTypes.FindNodeOrInsertPos(ID, InsertPos))
|
2007-10-11 04:55:41 +04:00
|
|
|
return QualType(QT, 0);
|
2009-09-09 19:08:12 +04:00
|
|
|
|
2010-05-15 15:32:37 +04:00
|
|
|
// Find the canonical object type.
|
2009-10-23 02:37:11 +04:00
|
|
|
QualType Canonical;
|
2010-05-15 15:32:37 +04:00
|
|
|
if (!ObjectT.isCanonical()) {
|
|
|
|
Canonical = getObjCObjectPointerType(getCanonicalType(ObjectT));
|
2009-10-23 02:37:11 +04:00
|
|
|
|
2010-05-15 15:32:37 +04:00
|
|
|
// Regenerate InsertPos.
|
|
|
|
ObjCObjectPointerTypes.FindNodeOrInsertPos(ID, InsertPos);
|
2009-10-23 02:37:11 +04:00
|
|
|
}
|
|
|
|
|
2010-05-15 15:32:37 +04:00
|
|
|
// No match.
|
|
|
|
void *Mem = Allocate(sizeof(ObjCObjectPointerType), TypeAlignment);
|
|
|
|
ObjCObjectPointerType *QType =
|
|
|
|
new (Mem) ObjCObjectPointerType(Canonical, ObjectT);
|
2009-09-29 23:42:55 +04:00
|
|
|
|
|
|
|
Types.push_back(QType);
|
2010-05-15 15:32:37 +04:00
|
|
|
ObjCObjectPointerTypes.InsertNode(QType, InsertPos);
|
2009-09-29 23:42:55 +04:00
|
|
|
return QualType(QType, 0);
|
|
|
|
}
|
|
|
|
|
2010-08-11 16:19:30 +04:00
|
|
|
/// getObjCInterfaceType - Return the unique reference to the type for the
|
|
|
|
/// specified ObjC interface decl. The list of protocols is optional.
|
2011-12-16 07:12:41 +04:00
|
|
|
QualType ASTContext::getObjCInterfaceType(const ObjCInterfaceDecl *Decl,
|
|
|
|
ObjCInterfaceDecl *PrevDecl) const {
|
2010-08-11 16:19:30 +04:00
|
|
|
if (Decl->TypeForDecl)
|
|
|
|
return QualType(Decl->TypeForDecl, 0);
|
2010-08-10 01:55:28 +04:00
|
|
|
|
2011-12-16 07:12:41 +04:00
|
|
|
if (PrevDecl) {
|
|
|
|
assert(PrevDecl->TypeForDecl && "previous decl has no TypeForDecl");
|
|
|
|
Decl->TypeForDecl = PrevDecl->TypeForDecl;
|
|
|
|
return QualType(PrevDecl->TypeForDecl, 0);
|
|
|
|
}
|
|
|
|
|
2011-12-16 20:34:57 +04:00
|
|
|
// Prefer the definition, if there is one.
|
|
|
|
if (const ObjCInterfaceDecl *Def = Decl->getDefinition())
|
|
|
|
Decl = Def;
|
|
|
|
|
2010-08-11 16:19:30 +04:00
|
|
|
void *Mem = Allocate(sizeof(ObjCInterfaceType), TypeAlignment);
|
|
|
|
ObjCInterfaceType *T = new (Mem) ObjCInterfaceType(Decl);
|
|
|
|
Decl->TypeForDecl = T;
|
|
|
|
Types.push_back(T);
|
|
|
|
return QualType(T, 0);
|
2010-05-15 15:32:37 +04:00
|
|
|
}
|
|
|
|
|
2009-02-27 02:50:07 +03:00
|
|
|
/// getTypeOfExprType - Unlike many "get<Type>" functions, we can't unique
|
|
|
|
/// TypeOfExprType AST's (since expression's are never shared). For example,
|
2007-08-01 22:02:17 +04:00
|
|
|
/// multiple declarations that refer to "typeof(x)" all contain different
|
2009-09-09 19:08:12 +04:00
|
|
|
/// DeclRefExpr's. This doesn't effect the type checker, since it operates
|
2007-08-01 22:02:17 +04:00
|
|
|
/// on canonical type's (which are always unique).
|
2011-01-12 12:06:06 +03:00
|
|
|
QualType ASTContext::getTypeOfExprType(Expr *tofExpr) const {
|
2009-07-08 04:03:05 +04:00
|
|
|
TypeOfExprType *toe;
|
2009-07-31 03:18:24 +04:00
|
|
|
if (tofExpr->isTypeDependent()) {
|
|
|
|
llvm::FoldingSetNodeID ID;
|
|
|
|
DependentTypeOfExprType::Profile(ID, *this, tofExpr);
|
2009-09-09 19:08:12 +04:00
|
|
|
|
2009-07-31 03:18:24 +04:00
|
|
|
void *InsertPos = 0;
|
|
|
|
DependentTypeOfExprType *Canon
|
|
|
|
= DependentTypeOfExprTypes.FindNodeOrInsertPos(ID, InsertPos);
|
|
|
|
if (Canon) {
|
|
|
|
// We already have a "canonical" version of an identical, dependent
|
|
|
|
// typeof(expr) type. Use that as our canonical type.
|
2009-09-25 03:30:46 +04:00
|
|
|
toe = new (*this, TypeAlignment) TypeOfExprType(tofExpr,
|
2009-07-31 03:18:24 +04:00
|
|
|
QualType((TypeOfExprType*)Canon, 0));
|
2011-08-18 03:08:45 +04:00
|
|
|
} else {
|
2009-07-31 03:18:24 +04:00
|
|
|
// Build a new, canonical typeof(expr) type.
|
2009-09-25 03:30:46 +04:00
|
|
|
Canon
|
|
|
|
= new (*this, TypeAlignment) DependentTypeOfExprType(*this, tofExpr);
|
2009-07-31 03:18:24 +04:00
|
|
|
DependentTypeOfExprTypes.InsertNode(Canon, InsertPos);
|
|
|
|
toe = Canon;
|
|
|
|
}
|
|
|
|
} else {
|
2009-07-08 04:03:05 +04:00
|
|
|
QualType Canonical = getCanonicalType(tofExpr->getType());
|
2009-09-25 03:30:46 +04:00
|
|
|
toe = new (*this, TypeAlignment) TypeOfExprType(tofExpr, Canonical);
|
2009-07-08 04:03:05 +04:00
|
|
|
}
|
2007-08-01 22:02:17 +04:00
|
|
|
Types.push_back(toe);
|
|
|
|
return QualType(toe, 0);
|
2007-07-31 16:34:36 +04:00
|
|
|
}
|
|
|
|
|
2007-08-01 22:02:17 +04:00
|
|
|
/// getTypeOfType - Unlike many "get<Type>" functions, we don't unique
|
|
|
|
/// TypeOfType AST's. The only motivation to unique these nodes would be
|
|
|
|
/// memory savings. Since typeof(t) is fairly uncommon, space shouldn't be
|
2009-09-09 19:08:12 +04:00
|
|
|
/// an issue. This doesn't effect the type checker, since it operates
|
2007-08-01 22:02:17 +04:00
|
|
|
/// on canonical type's (which are always unique).
|
2011-01-12 12:06:06 +03:00
|
|
|
QualType ASTContext::getTypeOfType(QualType tofType) const {
|
2008-04-07 02:59:24 +04:00
|
|
|
QualType Canonical = getCanonicalType(tofType);
|
2009-09-25 03:30:46 +04:00
|
|
|
TypeOfType *tot = new (*this, TypeAlignment) TypeOfType(tofType, Canonical);
|
2007-08-01 22:02:17 +04:00
|
|
|
Types.push_back(tot);
|
|
|
|
return QualType(tot, 0);
|
2007-07-31 16:34:36 +04:00
|
|
|
}
|
|
|
|
|
2009-06-25 01:24:56 +04:00
|
|
|
|
2009-06-24 23:06:50 +04:00
|
|
|
/// getDecltypeType - Unlike many "get<Type>" functions, we don't unique
|
|
|
|
/// DecltypeType AST's. The only motivation to unique these nodes would be
|
|
|
|
/// memory savings. Since decltype(t) is fairly uncommon, space shouldn't be
|
2009-09-09 19:08:12 +04:00
|
|
|
/// an issue. This doesn't effect the type checker, since it operates
|
2011-11-07 02:28:03 +04:00
|
|
|
/// on canonical types (which are always unique).
|
2012-02-12 22:42:33 +04:00
|
|
|
QualType ASTContext::getDecltypeType(Expr *e, QualType UnderlyingType) const {
|
2009-07-08 04:03:05 +04:00
|
|
|
DecltypeType *dt;
|
2011-07-01 05:22:09 +04:00
|
|
|
|
|
|
|
// C++0x [temp.type]p2:
|
|
|
|
// If an expression e involves a template parameter, decltype(e) denotes a
|
|
|
|
// unique dependent type. Two such decltype-specifiers refer to the same
|
|
|
|
// type only if their expressions are equivalent (14.5.6.1).
|
|
|
|
if (e->isInstantiationDependent()) {
|
2009-07-31 03:36:40 +04:00
|
|
|
llvm::FoldingSetNodeID ID;
|
|
|
|
DependentDecltypeType::Profile(ID, *this, e);
|
2009-09-09 19:08:12 +04:00
|
|
|
|
2009-07-31 03:36:40 +04:00
|
|
|
void *InsertPos = 0;
|
|
|
|
DependentDecltypeType *Canon
|
|
|
|
= DependentDecltypeTypes.FindNodeOrInsertPos(ID, InsertPos);
|
|
|
|
if (Canon) {
|
|
|
|
// We already have a "canonical" version of an equivalent, dependent
|
|
|
|
// decltype type. Use that as our canonical type.
|
2009-09-25 03:30:46 +04:00
|
|
|
dt = new (*this, TypeAlignment) DecltypeType(e, DependentTy,
|
2009-07-31 03:36:40 +04:00
|
|
|
QualType((DecltypeType*)Canon, 0));
|
2011-08-18 03:08:45 +04:00
|
|
|
} else {
|
2009-07-31 03:36:40 +04:00
|
|
|
// Build a new, canonical typeof(expr) type.
|
2009-09-25 03:30:46 +04:00
|
|
|
Canon = new (*this, TypeAlignment) DependentDecltypeType(*this, e);
|
2009-07-31 03:36:40 +04:00
|
|
|
DependentDecltypeTypes.InsertNode(Canon, InsertPos);
|
|
|
|
dt = Canon;
|
|
|
|
}
|
|
|
|
} else {
|
2012-02-12 22:42:33 +04:00
|
|
|
dt = new (*this, TypeAlignment) DecltypeType(e, UnderlyingType,
|
|
|
|
getCanonicalType(UnderlyingType));
|
2009-07-08 04:03:05 +04:00
|
|
|
}
|
2009-06-24 23:06:50 +04:00
|
|
|
Types.push_back(dt);
|
|
|
|
return QualType(dt, 0);
|
|
|
|
}
|
|
|
|
|
2011-05-25 02:41:36 +04:00
|
|
|
/// getUnaryTransformationType - We don't unique these, since the memory
|
|
|
|
/// savings are minimal and these are rare.
|
|
|
|
QualType ASTContext::getUnaryTransformType(QualType BaseType,
|
|
|
|
QualType UnderlyingType,
|
|
|
|
UnaryTransformType::UTTKind Kind)
|
|
|
|
const {
|
|
|
|
UnaryTransformType *Ty =
|
2011-05-25 21:51:54 +04:00
|
|
|
new (*this, TypeAlignment) UnaryTransformType (BaseType, UnderlyingType,
|
|
|
|
Kind,
|
|
|
|
UnderlyingType->isDependentType() ?
|
2012-03-05 20:02:06 +04:00
|
|
|
QualType() : getCanonicalType(UnderlyingType));
|
2011-05-25 02:41:36 +04:00
|
|
|
Types.push_back(Ty);
|
|
|
|
return QualType(Ty, 0);
|
|
|
|
}
|
|
|
|
|
2011-02-21 23:05:19 +03:00
|
|
|
/// getAutoType - We only unique auto types after they've been deduced.
|
2011-02-20 06:19:35 +03:00
|
|
|
QualType ASTContext::getAutoType(QualType DeducedType) const {
|
2011-02-21 23:05:19 +03:00
|
|
|
void *InsertPos = 0;
|
|
|
|
if (!DeducedType.isNull()) {
|
|
|
|
// Look in the folding set for an existing type.
|
|
|
|
llvm::FoldingSetNodeID ID;
|
|
|
|
AutoType::Profile(ID, DeducedType);
|
|
|
|
if (AutoType *AT = AutoTypes.FindNodeOrInsertPos(ID, InsertPos))
|
|
|
|
return QualType(AT, 0);
|
|
|
|
}
|
|
|
|
|
|
|
|
AutoType *AT = new (*this, TypeAlignment) AutoType(DeducedType);
|
|
|
|
Types.push_back(AT);
|
|
|
|
if (InsertPos)
|
|
|
|
AutoTypes.InsertNode(AT, InsertPos);
|
|
|
|
return QualType(AT, 0);
|
2011-02-20 06:19:35 +03:00
|
|
|
}
|
|
|
|
|
2011-10-07 03:00:33 +04:00
|
|
|
/// getAtomicType - Return the uniqued reference to the atomic type for
|
|
|
|
/// the given value type.
|
|
|
|
QualType ASTContext::getAtomicType(QualType T) const {
|
|
|
|
// Unique pointers, to guarantee there is only one pointer of a particular
|
|
|
|
// structure.
|
|
|
|
llvm::FoldingSetNodeID ID;
|
|
|
|
AtomicType::Profile(ID, T);
|
|
|
|
|
|
|
|
void *InsertPos = 0;
|
|
|
|
if (AtomicType *AT = AtomicTypes.FindNodeOrInsertPos(ID, InsertPos))
|
|
|
|
return QualType(AT, 0);
|
|
|
|
|
|
|
|
// If the atomic value type isn't canonical, this won't be a canonical type
|
|
|
|
// either, so fill in the canonical type field.
|
|
|
|
QualType Canonical;
|
|
|
|
if (!T.isCanonical()) {
|
|
|
|
Canonical = getAtomicType(getCanonicalType(T));
|
|
|
|
|
|
|
|
// Get the new insert position for the node we care about.
|
|
|
|
AtomicType *NewIP = AtomicTypes.FindNodeOrInsertPos(ID, InsertPos);
|
|
|
|
assert(NewIP == 0 && "Shouldn't be in the map!"); (void)NewIP;
|
|
|
|
}
|
|
|
|
AtomicType *New = new (*this, TypeAlignment) AtomicType(T, Canonical);
|
|
|
|
Types.push_back(New);
|
|
|
|
AtomicTypes.InsertNode(New, InsertPos);
|
|
|
|
return QualType(New, 0);
|
|
|
|
}
|
|
|
|
|
2011-04-15 02:09:26 +04:00
|
|
|
/// getAutoDeductType - Get type pattern for deducing against 'auto'.
|
|
|
|
QualType ASTContext::getAutoDeductType() const {
|
|
|
|
if (AutoDeductTy.isNull())
|
|
|
|
AutoDeductTy = getAutoType(QualType());
|
|
|
|
assert(!AutoDeductTy.isNull() && "can't build 'auto' pattern");
|
|
|
|
return AutoDeductTy;
|
|
|
|
}
|
|
|
|
|
|
|
|
/// getAutoRRefDeductType - Get type pattern for deducing against 'auto &&'.
|
|
|
|
QualType ASTContext::getAutoRRefDeductType() const {
|
|
|
|
if (AutoRRefDeductTy.isNull())
|
|
|
|
AutoRRefDeductTy = getRValueReferenceType(getAutoDeductType());
|
|
|
|
assert(!AutoRRefDeductTy.isNull() && "can't build 'auto &&' pattern");
|
|
|
|
return AutoRRefDeductTy;
|
|
|
|
}
|
|
|
|
|
2007-07-11 21:01:13 +04:00
|
|
|
/// getTagDeclType - Return the unique reference to the type for the
|
|
|
|
/// specified TagDecl (struct/union/class/enum) decl.
|
2011-01-12 12:06:06 +03:00
|
|
|
QualType ASTContext::getTagDeclType(const TagDecl *Decl) const {
|
2007-11-27 00:16:01 +03:00
|
|
|
assert (Decl);
|
2009-08-07 22:05:12 +04:00
|
|
|
// FIXME: What is the design on getTagDeclType when it requires casting
|
|
|
|
// away const? mutable?
|
|
|
|
return getTypeDeclType(const_cast<TagDecl*>(Decl));
|
2007-07-11 21:01:13 +04:00
|
|
|
}
|
|
|
|
|
2009-09-09 19:08:12 +04:00
|
|
|
/// getSizeType - Return the unique type for "size_t" (C99 7.17), the result
|
|
|
|
/// of the sizeof operator (C99 6.5.3.4p4). The value is target dependent and
|
|
|
|
/// needs to agree with the definition in <stddef.h>.
|
2009-12-12 03:26:23 +03:00
|
|
|
CanQualType ASTContext::getSizeType() const {
|
2011-09-02 04:18:52 +04:00
|
|
|
return getFromTargetType(Target->getSizeType());
|
2007-07-11 21:01:13 +04:00
|
|
|
}
|
|
|
|
|
2011-10-27 12:29:09 +04:00
|
|
|
/// getIntMaxType - Return the unique type for "intmax_t" (C99 7.18.1.5).
|
|
|
|
CanQualType ASTContext::getIntMaxType() const {
|
|
|
|
return getFromTargetType(Target->getIntMaxType());
|
|
|
|
}
|
|
|
|
|
|
|
|
/// getUIntMaxType - Return the unique type for "uintmax_t" (C99 7.18.1.5).
|
|
|
|
CanQualType ASTContext::getUIntMaxType() const {
|
|
|
|
return getFromTargetType(Target->getUIntMaxType());
|
|
|
|
}
|
|
|
|
|
2008-08-09 20:51:54 +04:00
|
|
|
/// getSignedWCharType - Return the type of "signed wchar_t".
|
|
|
|
/// Used when in C++, as a GCC extension.
|
|
|
|
QualType ASTContext::getSignedWCharType() const {
|
|
|
|
// FIXME: derive from "Target" ?
|
|
|
|
return WCharTy;
|
|
|
|
}
|
|
|
|
|
|
|
|
/// getUnsignedWCharType - Return the type of "unsigned wchar_t".
|
|
|
|
/// Used when in C++, as a GCC extension.
|
|
|
|
QualType ASTContext::getUnsignedWCharType() const {
|
|
|
|
// FIXME: derive from "Target" ?
|
|
|
|
return UnsignedIntTy;
|
|
|
|
}
|
|
|
|
|
2011-10-27 12:29:09 +04:00
|
|
|
/// getPointerDiffType - Return the unique type for "ptrdiff_t" (C99 7.17)
|
2007-07-13 07:05:23 +04:00
|
|
|
/// defined in <stddef.h>. Pointer - pointer requires this (C99 6.5.6p9).
|
|
|
|
QualType ASTContext::getPointerDiffType() const {
|
2011-09-02 04:18:52 +04:00
|
|
|
return getFromTargetType(Target->getPtrDiffType(0));
|
2007-07-13 07:05:23 +04:00
|
|
|
}
|
|
|
|
|
2008-04-02 09:18:44 +04:00
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
// Type Operators
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
|
2011-01-12 12:06:06 +03:00
|
|
|
CanQualType ASTContext::getCanonicalParamType(QualType T) const {
|
2009-10-23 02:37:11 +04:00
|
|
|
// Push qualifiers into arrays, and then discard any remaining
|
|
|
|
// qualifiers.
|
|
|
|
T = getCanonicalType(T);
|
2010-09-24 21:30:16 +04:00
|
|
|
T = getVariableArrayDecayedType(T);
|
2009-10-23 02:37:11 +04:00
|
|
|
const Type *Ty = T.getTypePtr();
|
|
|
|
QualType Result;
|
|
|
|
if (isa<ArrayType>(Ty)) {
|
|
|
|
Result = getArrayDecayedType(QualType(Ty,0));
|
|
|
|
} else if (isa<FunctionType>(Ty)) {
|
|
|
|
Result = getPointerType(QualType(Ty, 0));
|
|
|
|
} else {
|
|
|
|
Result = QualType(Ty, 0);
|
|
|
|
}
|
|
|
|
|
|
|
|
return CanQualType::CreateUnsafe(Result);
|
|
|
|
}
|
|
|
|
|
2011-01-18 10:41:22 +03:00
|
|
|
QualType ASTContext::getUnqualifiedArrayType(QualType type,
|
|
|
|
Qualifiers &quals) {
|
|
|
|
SplitQualType splitType = type.getSplitUnqualifiedType();
|
|
|
|
|
|
|
|
// FIXME: getSplitUnqualifiedType() actually walks all the way to
|
|
|
|
// the unqualified desugared type and then drops it on the floor.
|
|
|
|
// We then have to strip that sugar back off with
|
|
|
|
// getUnqualifiedDesugaredType(), which is silly.
|
|
|
|
const ArrayType *AT =
|
2012-02-08 04:46:36 +04:00
|
|
|
dyn_cast<ArrayType>(splitType.Ty->getUnqualifiedDesugaredType());
|
2011-01-18 10:41:22 +03:00
|
|
|
|
|
|
|
// If we don't have an array, just use the results in splitType.
|
2010-05-17 22:45:21 +04:00
|
|
|
if (!AT) {
|
2012-02-08 04:46:36 +04:00
|
|
|
quals = splitType.Quals;
|
|
|
|
return QualType(splitType.Ty, 0);
|
2009-12-29 10:16:59 +03:00
|
|
|
}
|
|
|
|
|
2011-01-18 10:41:22 +03:00
|
|
|
// Otherwise, recurse on the array's element type.
|
|
|
|
QualType elementType = AT->getElementType();
|
|
|
|
QualType unqualElementType = getUnqualifiedArrayType(elementType, quals);
|
|
|
|
|
|
|
|
// If that didn't change the element type, AT has no qualifiers, so we
|
|
|
|
// can just use the results in splitType.
|
|
|
|
if (elementType == unqualElementType) {
|
|
|
|
assert(quals.empty()); // from the recursive call
|
2012-02-08 04:46:36 +04:00
|
|
|
quals = splitType.Quals;
|
|
|
|
return QualType(splitType.Ty, 0);
|
2011-01-18 10:41:22 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
// Otherwise, add in the qualifiers from the outermost type, then
|
|
|
|
// build the type back up.
|
2012-02-08 04:46:36 +04:00
|
|
|
quals.addConsistentQualifiers(splitType.Quals);
|
2009-12-29 10:16:59 +03:00
|
|
|
|
2010-05-17 22:45:21 +04:00
|
|
|
if (const ConstantArrayType *CAT = dyn_cast<ConstantArrayType>(AT)) {
|
2011-01-18 10:41:22 +03:00
|
|
|
return getConstantArrayType(unqualElementType, CAT->getSize(),
|
2009-12-29 10:16:59 +03:00
|
|
|
CAT->getSizeModifier(), 0);
|
|
|
|
}
|
|
|
|
|
2010-05-17 22:45:21 +04:00
|
|
|
if (const IncompleteArrayType *IAT = dyn_cast<IncompleteArrayType>(AT)) {
|
2011-01-18 10:41:22 +03:00
|
|
|
return getIncompleteArrayType(unqualElementType, IAT->getSizeModifier(), 0);
|
2009-12-29 10:16:59 +03:00
|
|
|
}
|
|
|
|
|
2010-05-17 22:45:21 +04:00
|
|
|
if (const VariableArrayType *VAT = dyn_cast<VariableArrayType>(AT)) {
|
2011-01-18 10:41:22 +03:00
|
|
|
return getVariableArrayType(unqualElementType,
|
2010-10-26 11:05:15 +04:00
|
|
|
VAT->getSizeExpr(),
|
2010-05-17 22:45:21 +04:00
|
|
|
VAT->getSizeModifier(),
|
|
|
|
VAT->getIndexTypeCVRQualifiers(),
|
|
|
|
VAT->getBracketsRange());
|
|
|
|
}
|
|
|
|
|
|
|
|
const DependentSizedArrayType *DSAT = cast<DependentSizedArrayType>(AT);
|
2011-01-18 10:41:22 +03:00
|
|
|
return getDependentSizedArrayType(unqualElementType, DSAT->getSizeExpr(),
|
2009-12-29 10:16:59 +03:00
|
|
|
DSAT->getSizeModifier(), 0,
|
|
|
|
SourceRange());
|
|
|
|
}
|
|
|
|
|
2010-06-09 07:53:18 +04:00
|
|
|
/// UnwrapSimilarPointerTypes - If T1 and T2 are pointer types that
|
|
|
|
/// may be similar (C++ 4.4), replaces T1 and T2 with the type that
|
|
|
|
/// they point to and return true. If T1 and T2 aren't pointer types
|
|
|
|
/// or pointer-to-member types, or if they are not similar at this
|
|
|
|
/// level, returns false and leaves T1 and T2 unchanged. Top-level
|
|
|
|
/// qualifiers on T1 and T2 are ignored. This function will typically
|
|
|
|
/// be called in a loop that successively "unwraps" pointer and
|
|
|
|
/// pointer-to-member types to compare them at each level.
|
|
|
|
bool ASTContext::UnwrapSimilarPointerTypes(QualType &T1, QualType &T2) {
|
|
|
|
const PointerType *T1PtrType = T1->getAs<PointerType>(),
|
|
|
|
*T2PtrType = T2->getAs<PointerType>();
|
|
|
|
if (T1PtrType && T2PtrType) {
|
|
|
|
T1 = T1PtrType->getPointeeType();
|
|
|
|
T2 = T2PtrType->getPointeeType();
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
const MemberPointerType *T1MPType = T1->getAs<MemberPointerType>(),
|
|
|
|
*T2MPType = T2->getAs<MemberPointerType>();
|
|
|
|
if (T1MPType && T2MPType &&
|
|
|
|
hasSameUnqualifiedType(QualType(T1MPType->getClass(), 0),
|
|
|
|
QualType(T2MPType->getClass(), 0))) {
|
|
|
|
T1 = T1MPType->getPointeeType();
|
|
|
|
T2 = T2MPType->getPointeeType();
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2012-03-11 11:00:24 +04:00
|
|
|
if (getLangOpts().ObjC1) {
|
2010-06-09 07:53:18 +04:00
|
|
|
const ObjCObjectPointerType *T1OPType = T1->getAs<ObjCObjectPointerType>(),
|
|
|
|
*T2OPType = T2->getAs<ObjCObjectPointerType>();
|
|
|
|
if (T1OPType && T2OPType) {
|
|
|
|
T1 = T1OPType->getPointeeType();
|
|
|
|
T2 = T2OPType->getPointeeType();
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// FIXME: Block pointers, too?
|
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2011-01-12 12:06:06 +03:00
|
|
|
DeclarationNameInfo
|
|
|
|
ASTContext::getNameForTemplate(TemplateName Name,
|
|
|
|
SourceLocation NameLoc) const {
|
2011-06-30 12:33:18 +04:00
|
|
|
switch (Name.getKind()) {
|
|
|
|
case TemplateName::QualifiedTemplate:
|
|
|
|
case TemplateName::Template:
|
2010-08-12 02:01:17 +04:00
|
|
|
// DNInfo work in progress: CHECKME: what about DNLoc?
|
2011-06-30 12:33:18 +04:00
|
|
|
return DeclarationNameInfo(Name.getAsTemplateDecl()->getDeclName(),
|
|
|
|
NameLoc);
|
2010-08-12 02:01:17 +04:00
|
|
|
|
2011-06-30 12:33:18 +04:00
|
|
|
case TemplateName::OverloadedTemplate: {
|
|
|
|
OverloadedTemplateStorage *Storage = Name.getAsOverloadedTemplate();
|
|
|
|
// DNInfo work in progress: CHECKME: what about DNLoc?
|
|
|
|
return DeclarationNameInfo((*Storage->begin())->getDeclName(), NameLoc);
|
|
|
|
}
|
|
|
|
|
|
|
|
case TemplateName::DependentTemplate: {
|
|
|
|
DependentTemplateName *DTN = Name.getAsDependentTemplateName();
|
2010-08-12 02:01:17 +04:00
|
|
|
DeclarationName DName;
|
2009-11-24 21:42:40 +03:00
|
|
|
if (DTN->isIdentifier()) {
|
2010-08-12 02:01:17 +04:00
|
|
|
DName = DeclarationNames.getIdentifier(DTN->getIdentifier());
|
|
|
|
return DeclarationNameInfo(DName, NameLoc);
|
2009-11-24 21:42:40 +03:00
|
|
|
} else {
|
2010-08-12 02:01:17 +04:00
|
|
|
DName = DeclarationNames.getCXXOperatorName(DTN->getOperator());
|
|
|
|
// DNInfo work in progress: FIXME: source locations?
|
|
|
|
DeclarationNameLoc DNLoc;
|
|
|
|
DNLoc.CXXOperatorName.BeginOpNameLoc = SourceLocation().getRawEncoding();
|
|
|
|
DNLoc.CXXOperatorName.EndOpNameLoc = SourceLocation().getRawEncoding();
|
|
|
|
return DeclarationNameInfo(DName, NameLoc, DNLoc);
|
2009-11-24 21:42:40 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-06-30 12:33:18 +04:00
|
|
|
case TemplateName::SubstTemplateTemplateParm: {
|
|
|
|
SubstTemplateTemplateParmStorage *subst
|
|
|
|
= Name.getAsSubstTemplateTemplateParm();
|
|
|
|
return DeclarationNameInfo(subst->getParameter()->getDeclName(),
|
|
|
|
NameLoc);
|
|
|
|
}
|
|
|
|
|
|
|
|
case TemplateName::SubstTemplateTemplateParmPack: {
|
|
|
|
SubstTemplateTemplateParmPackStorage *subst
|
|
|
|
= Name.getAsSubstTemplateTemplateParmPack();
|
|
|
|
return DeclarationNameInfo(subst->getParameterPack()->getDeclName(),
|
|
|
|
NameLoc);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
llvm_unreachable("bad template name kind!");
|
2009-11-24 21:42:40 +03:00
|
|
|
}
|
|
|
|
|
2011-01-12 12:06:06 +03:00
|
|
|
TemplateName ASTContext::getCanonicalTemplateName(TemplateName Name) const {
|
2011-06-30 12:33:18 +04:00
|
|
|
switch (Name.getKind()) {
|
|
|
|
case TemplateName::QualifiedTemplate:
|
|
|
|
case TemplateName::Template: {
|
|
|
|
TemplateDecl *Template = Name.getAsTemplateDecl();
|
2010-06-17 01:09:37 +04:00
|
|
|
if (TemplateTemplateParmDecl *TTP
|
2011-06-30 12:33:18 +04:00
|
|
|
= dyn_cast<TemplateTemplateParmDecl>(Template))
|
2010-06-17 01:09:37 +04:00
|
|
|
Template = getCanonicalTemplateTemplateParmDecl(TTP);
|
|
|
|
|
|
|
|
// The canonical template name is the canonical template declaration.
|
2009-07-18 04:34:25 +04:00
|
|
|
return TemplateName(cast<TemplateDecl>(Template->getCanonicalDecl()));
|
2010-06-17 01:09:37 +04:00
|
|
|
}
|
2009-05-07 10:41:52 +04:00
|
|
|
|
2011-06-30 12:33:18 +04:00
|
|
|
case TemplateName::OverloadedTemplate:
|
|
|
|
llvm_unreachable("cannot canonicalize overloaded template");
|
|
|
|
|
|
|
|
case TemplateName::DependentTemplate: {
|
|
|
|
DependentTemplateName *DTN = Name.getAsDependentTemplateName();
|
|
|
|
assert(DTN && "Non-dependent template names must refer to template decls.");
|
|
|
|
return DTN->CanonicalTemplateName;
|
|
|
|
}
|
|
|
|
|
|
|
|
case TemplateName::SubstTemplateTemplateParm: {
|
|
|
|
SubstTemplateTemplateParmStorage *subst
|
|
|
|
= Name.getAsSubstTemplateTemplateParm();
|
|
|
|
return getCanonicalTemplateName(subst->getReplacement());
|
2011-01-15 09:45:20 +03:00
|
|
|
}
|
2009-09-09 19:08:12 +04:00
|
|
|
|
2011-06-30 12:33:18 +04:00
|
|
|
case TemplateName::SubstTemplateTemplateParmPack: {
|
|
|
|
SubstTemplateTemplateParmPackStorage *subst
|
|
|
|
= Name.getAsSubstTemplateTemplateParmPack();
|
|
|
|
TemplateTemplateParmDecl *canonParameter
|
|
|
|
= getCanonicalTemplateTemplateParmDecl(subst->getParameterPack());
|
|
|
|
TemplateArgument canonArgPack
|
|
|
|
= getCanonicalTemplateArgument(subst->getArgumentPack());
|
|
|
|
return getSubstTemplateTemplateParmPack(canonParameter, canonArgPack);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
llvm_unreachable("bad template name!");
|
2009-05-07 10:41:52 +04:00
|
|
|
}
|
|
|
|
|
2009-11-12 02:06:43 +03:00
|
|
|
bool ASTContext::hasSameTemplateName(TemplateName X, TemplateName Y) {
|
|
|
|
X = getCanonicalTemplateName(X);
|
|
|
|
Y = getCanonicalTemplateName(Y);
|
|
|
|
return X.getAsVoidPointer() == Y.getAsVoidPointer();
|
|
|
|
}
|
|
|
|
|
2009-09-09 19:08:12 +04:00
|
|
|
TemplateArgument
|
2011-01-12 12:06:06 +03:00
|
|
|
ASTContext::getCanonicalTemplateArgument(const TemplateArgument &Arg) const {
|
2009-07-29 03:00:59 +04:00
|
|
|
switch (Arg.getKind()) {
|
|
|
|
case TemplateArgument::Null:
|
|
|
|
return Arg;
|
2009-09-09 19:08:12 +04:00
|
|
|
|
2009-07-29 03:00:59 +04:00
|
|
|
case TemplateArgument::Expression:
|
|
|
|
return Arg;
|
2009-09-09 19:08:12 +04:00
|
|
|
|
2009-07-29 03:00:59 +04:00
|
|
|
case TemplateArgument::Declaration:
|
2009-10-29 11:12:44 +03:00
|
|
|
return TemplateArgument(Arg.getAsDecl()->getCanonicalDecl());
|
2009-09-09 19:08:12 +04:00
|
|
|
|
2009-11-11 04:00:40 +03:00
|
|
|
case TemplateArgument::Template:
|
|
|
|
return TemplateArgument(getCanonicalTemplateName(Arg.getAsTemplate()));
|
2011-01-05 21:58:31 +03:00
|
|
|
|
|
|
|
case TemplateArgument::TemplateExpansion:
|
|
|
|
return TemplateArgument(getCanonicalTemplateName(
|
|
|
|
Arg.getAsTemplateOrTemplatePattern()),
|
2011-01-15 02:41:42 +03:00
|
|
|
Arg.getNumTemplateExpansions());
|
2011-01-05 21:58:31 +03:00
|
|
|
|
2009-07-29 03:00:59 +04:00
|
|
|
case TemplateArgument::Integral:
|
2009-10-29 11:12:44 +03:00
|
|
|
return TemplateArgument(*Arg.getAsIntegral(),
|
2009-07-29 03:00:59 +04:00
|
|
|
getCanonicalType(Arg.getIntegralType()));
|
2009-09-09 19:08:12 +04:00
|
|
|
|
2009-07-29 03:00:59 +04:00
|
|
|
case TemplateArgument::Type:
|
2009-10-29 11:12:44 +03:00
|
|
|
return TemplateArgument(getCanonicalType(Arg.getAsType()));
|
2009-09-09 19:08:12 +04:00
|
|
|
|
2009-07-29 03:00:59 +04:00
|
|
|
case TemplateArgument::Pack: {
|
2010-12-20 19:52:59 +03:00
|
|
|
if (Arg.pack_size() == 0)
|
|
|
|
return Arg;
|
|
|
|
|
2010-11-08 02:05:16 +03:00
|
|
|
TemplateArgument *CanonArgs
|
|
|
|
= new (*this) TemplateArgument[Arg.pack_size()];
|
2009-07-29 03:00:59 +04:00
|
|
|
unsigned Idx = 0;
|
2009-09-09 19:08:12 +04:00
|
|
|
for (TemplateArgument::pack_iterator A = Arg.pack_begin(),
|
2009-07-29 03:00:59 +04:00
|
|
|
AEnd = Arg.pack_end();
|
|
|
|
A != AEnd; (void)++A, ++Idx)
|
|
|
|
CanonArgs[Idx] = getCanonicalTemplateArgument(*A);
|
2009-09-09 19:08:12 +04:00
|
|
|
|
2010-11-08 02:05:16 +03:00
|
|
|
return TemplateArgument(CanonArgs, Arg.pack_size());
|
2009-07-29 03:00:59 +04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Silence GCC warning
|
2011-09-23 09:06:16 +04:00
|
|
|
llvm_unreachable("Unhandled template argument kind");
|
2009-07-29 03:00:59 +04:00
|
|
|
}
|
|
|
|
|
2009-03-28 02:10:48 +03:00
|
|
|
NestedNameSpecifier *
|
2011-01-12 12:06:06 +03:00
|
|
|
ASTContext::getCanonicalNestedNameSpecifier(NestedNameSpecifier *NNS) const {
|
2009-09-09 19:08:12 +04:00
|
|
|
if (!NNS)
|
2009-03-28 02:10:48 +03:00
|
|
|
return 0;
|
|
|
|
|
|
|
|
switch (NNS->getKind()) {
|
|
|
|
case NestedNameSpecifier::Identifier:
|
|
|
|
// Canonicalize the prefix but keep the identifier the same.
|
2009-09-09 19:08:12 +04:00
|
|
|
return NestedNameSpecifier::Create(*this,
|
2009-03-28 02:10:48 +03:00
|
|
|
getCanonicalNestedNameSpecifier(NNS->getPrefix()),
|
|
|
|
NNS->getAsIdentifier());
|
|
|
|
|
|
|
|
case NestedNameSpecifier::Namespace:
|
|
|
|
// A namespace is canonical; build a nested-name-specifier with
|
|
|
|
// this namespace and no prefix.
|
2011-02-24 05:36:08 +03:00
|
|
|
return NestedNameSpecifier::Create(*this, 0,
|
|
|
|
NNS->getAsNamespace()->getOriginalNamespace());
|
|
|
|
|
|
|
|
case NestedNameSpecifier::NamespaceAlias:
|
|
|
|
// A namespace is canonical; build a nested-name-specifier with
|
|
|
|
// this namespace and no prefix.
|
|
|
|
return NestedNameSpecifier::Create(*this, 0,
|
|
|
|
NNS->getAsNamespaceAlias()->getNamespace()
|
|
|
|
->getOriginalNamespace());
|
2009-03-28 02:10:48 +03:00
|
|
|
|
|
|
|
case NestedNameSpecifier::TypeSpec:
|
|
|
|
case NestedNameSpecifier::TypeSpecWithTemplate: {
|
|
|
|
QualType T = getCanonicalType(QualType(NNS->getAsType(), 0));
|
2010-11-04 03:09:33 +03:00
|
|
|
|
|
|
|
// If we have some kind of dependent-named type (e.g., "typename T::type"),
|
|
|
|
// break it apart into its prefix and identifier, then reconsititute those
|
|
|
|
// as the canonical nested-name-specifier. This is required to canonicalize
|
|
|
|
// a dependent nested-name-specifier involving typedefs of dependent-name
|
|
|
|
// types, e.g.,
|
|
|
|
// typedef typename T::type T1;
|
|
|
|
// typedef typename T1::type T2;
|
2012-03-03 08:09:56 +04:00
|
|
|
if (const DependentNameType *DNT = T->getAs<DependentNameType>())
|
|
|
|
return NestedNameSpecifier::Create(*this, DNT->getQualifier(),
|
2010-11-04 03:09:33 +03:00
|
|
|
const_cast<IdentifierInfo *>(DNT->getIdentifier()));
|
|
|
|
|
2012-03-03 08:09:56 +04:00
|
|
|
// Otherwise, just canonicalize the type, and force it to be a TypeSpec.
|
|
|
|
// FIXME: Why are TypeSpec and TypeSpecWithTemplate distinct in the
|
|
|
|
// first place?
|
2011-01-19 13:06:00 +03:00
|
|
|
return NestedNameSpecifier::Create(*this, 0, false,
|
|
|
|
const_cast<Type*>(T.getTypePtr()));
|
2009-03-28 02:10:48 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
case NestedNameSpecifier::Global:
|
|
|
|
// The global specifier is canonical and unique.
|
|
|
|
return NNS;
|
|
|
|
}
|
|
|
|
|
2012-01-17 10:56:22 +04:00
|
|
|
llvm_unreachable("Invalid NestedNameSpecifier::Kind!");
|
2009-03-28 02:10:48 +03:00
|
|
|
}
|
|
|
|
|
2008-08-04 11:31:14 +04:00
|
|
|
|
2011-01-12 12:06:06 +03:00
|
|
|
const ArrayType *ASTContext::getAsArrayType(QualType T) const {
|
2008-08-04 11:31:14 +04:00
|
|
|
// Handle the non-qualified case efficiently.
|
First part of changes to eliminate problems with cv-qualifiers and
sugared types. The basic problem is that our qualifier accessors
(getQualifiers, getCVRQualifiers, isConstQualified, etc.) only look at
the current QualType and not at any qualifiers that come from sugared
types, meaning that we won't see these qualifiers through, e.g.,
typedefs:
typedef const int CInt;
typedef CInt Self;
Self.isConstQualified() currently returns false!
Various bugs (e.g., PR5383) have cropped up all over the front end due
to such problems. I'm addressing this problem by splitting each
qualifier accessor into two versions:
- the "local" version only returns qualifiers on this particular
QualType instance
- the "normal" version that will eventually combine qualifiers from this
QualType instance with the qualifiers on the canonical type to
produce the full set of qualifiers.
This commit adds the local versions and switches a few callers from
the "normal" version (e.g., isConstQualified) over to the "local"
version (e.g., isLocalConstQualified) when that is the right thing to
do, e.g., because we're printing or serializing the qualifiers. Also,
switch a bunch of
Context.getCanonicalType(T1).getUnqualifiedType() == Context.getCanonicalType(T2).getQualifiedType()
expressions over to
Context.hasSameUnqualifiedType(T1, T2)
git-svn-id: https://llvm.org/svn/llvm-project/cfe/trunk@88969 91177308-0d34-0410-b5e6-96231b3b80d8
2009-11-17 00:35:15 +03:00
|
|
|
if (!T.hasLocalQualifiers()) {
|
2008-08-04 11:31:14 +04:00
|
|
|
// Handle the common positive case fast.
|
|
|
|
if (const ArrayType *AT = dyn_cast<ArrayType>(T))
|
|
|
|
return AT;
|
|
|
|
}
|
2009-09-09 19:08:12 +04:00
|
|
|
|
2009-09-24 23:53:00 +04:00
|
|
|
// Handle the common negative case fast.
|
2011-01-19 13:06:00 +03:00
|
|
|
if (!isa<ArrayType>(T.getCanonicalType()))
|
2008-08-04 11:31:14 +04:00
|
|
|
return 0;
|
2009-09-09 19:08:12 +04:00
|
|
|
|
2009-09-24 23:53:00 +04:00
|
|
|
// Apply any qualifiers from the array type to the element type. This
|
2008-08-04 11:31:14 +04:00
|
|
|
// implements C99 6.7.3p8: "If the specification of an array type includes
|
|
|
|
// any type qualifiers, the element type is so qualified, not the array type."
|
2009-09-09 19:08:12 +04:00
|
|
|
|
2008-08-04 11:31:14 +04:00
|
|
|
// If we get here, we either have type qualifiers on the type, or we have
|
|
|
|
// sugar such as a typedef in the way. If we have type qualifiers on the type
|
2009-08-05 09:36:45 +04:00
|
|
|
// we must propagate them down into the element type.
|
2009-09-24 23:53:00 +04:00
|
|
|
|
2011-01-19 13:06:00 +03:00
|
|
|
SplitQualType split = T.getSplitDesugaredType();
|
2012-02-08 04:46:36 +04:00
|
|
|
Qualifiers qs = split.Quals;
|
2009-09-09 19:08:12 +04:00
|
|
|
|
2008-08-04 11:31:14 +04:00
|
|
|
// If we have a simple case, just return now.
|
2012-02-08 04:46:36 +04:00
|
|
|
const ArrayType *ATy = dyn_cast<ArrayType>(split.Ty);
|
2011-01-19 13:06:00 +03:00
|
|
|
if (ATy == 0 || qs.empty())
|
2008-08-04 11:31:14 +04:00
|
|
|
return ATy;
|
2009-09-09 19:08:12 +04:00
|
|
|
|
2008-08-04 11:31:14 +04:00
|
|
|
// Otherwise, we have an array and we have qualifiers on it. Push the
|
|
|
|
// qualifiers into the array element type and return a new array type.
|
2011-01-19 13:06:00 +03:00
|
|
|
QualType NewEltTy = getQualifiedType(ATy->getElementType(), qs);
|
2009-09-09 19:08:12 +04:00
|
|
|
|
2008-08-04 11:31:14 +04:00
|
|
|
if (const ConstantArrayType *CAT = dyn_cast<ConstantArrayType>(ATy))
|
|
|
|
return cast<ArrayType>(getConstantArrayType(NewEltTy, CAT->getSize(),
|
|
|
|
CAT->getSizeModifier(),
|
2009-09-24 23:53:00 +04:00
|
|
|
CAT->getIndexTypeCVRQualifiers()));
|
2008-08-04 11:31:14 +04:00
|
|
|
if (const IncompleteArrayType *IAT = dyn_cast<IncompleteArrayType>(ATy))
|
|
|
|
return cast<ArrayType>(getIncompleteArrayType(NewEltTy,
|
|
|
|
IAT->getSizeModifier(),
|
2009-09-24 23:53:00 +04:00
|
|
|
IAT->getIndexTypeCVRQualifiers()));
|
2008-12-06 02:32:09 +03:00
|
|
|
|
2009-09-09 19:08:12 +04:00
|
|
|
if (const DependentSizedArrayType *DSAT
|
2008-12-06 02:32:09 +03:00
|
|
|
= dyn_cast<DependentSizedArrayType>(ATy))
|
|
|
|
return cast<ArrayType>(
|
2009-09-09 19:08:12 +04:00
|
|
|
getDependentSizedArrayType(NewEltTy,
|
2010-10-26 11:05:15 +04:00
|
|
|
DSAT->getSizeExpr(),
|
2008-12-06 02:32:09 +03:00
|
|
|
DSAT->getSizeModifier(),
|
2009-09-24 23:53:00 +04:00
|
|
|
DSAT->getIndexTypeCVRQualifiers(),
|
2009-07-06 19:59:29 +04:00
|
|
|
DSAT->getBracketsRange()));
|
2009-09-09 19:08:12 +04:00
|
|
|
|
2008-08-04 11:31:14 +04:00
|
|
|
const VariableArrayType *VAT = cast<VariableArrayType>(ATy);
|
2009-07-06 19:59:29 +04:00
|
|
|
return cast<ArrayType>(getVariableArrayType(NewEltTy,
|
2010-10-26 11:05:15 +04:00
|
|
|
VAT->getSizeExpr(),
|
2008-08-04 11:31:14 +04:00
|
|
|
VAT->getSizeModifier(),
|
2009-09-24 23:53:00 +04:00
|
|
|
VAT->getIndexTypeCVRQualifiers(),
|
2009-07-06 19:59:29 +04:00
|
|
|
VAT->getBracketsRange()));
|
2008-04-07 02:41:35 +04:00
|
|
|
}
|
|
|
|
|
2011-07-12 08:42:08 +04:00
|
|
|
QualType ASTContext::getAdjustedParameterType(QualType T) {
|
|
|
|
// C99 6.7.5.3p7:
|
|
|
|
// A declaration of a parameter as "array of type" shall be
|
|
|
|
// adjusted to "qualified pointer to type", where the type
|
|
|
|
// qualifiers (if any) are those specified within the [ and ] of
|
|
|
|
// the array type derivation.
|
|
|
|
if (T->isArrayType())
|
|
|
|
return getArrayDecayedType(T);
|
|
|
|
|
|
|
|
// C99 6.7.5.3p8:
|
|
|
|
// A declaration of a parameter as "function returning type"
|
|
|
|
// shall be adjusted to "pointer to function returning type", as
|
|
|
|
// in 6.3.2.1.
|
|
|
|
if (T->isFunctionType())
|
|
|
|
return getPointerType(T);
|
|
|
|
|
|
|
|
return T;
|
|
|
|
}
|
|
|
|
|
|
|
|
QualType ASTContext::getSignatureParameterType(QualType T) {
|
|
|
|
T = getVariableArrayDecayedType(T);
|
|
|
|
T = getAdjustedParameterType(T);
|
|
|
|
return T.getUnqualifiedType();
|
|
|
|
}
|
|
|
|
|
2008-04-02 09:18:44 +04:00
|
|
|
/// getArrayDecayedType - Return the properly qualified result of decaying the
|
|
|
|
/// specified array type to a pointer. This operation is non-trivial when
|
|
|
|
/// handling typedefs etc. The canonical type of "T" must be an array type,
|
|
|
|
/// this returns a pointer to a properly qualified element of the array.
|
|
|
|
///
|
|
|
|
/// See C99 6.7.5.3p7 and C99 6.3.2.1p3.
|
2011-01-12 12:06:06 +03:00
|
|
|
QualType ASTContext::getArrayDecayedType(QualType Ty) const {
|
2008-08-04 11:31:14 +04:00
|
|
|
// Get the element type with 'getAsArrayType' so that we don't lose any
|
|
|
|
// typedefs in the element type of the array. This also handles propagation
|
|
|
|
// of type qualifiers from the array type into the element type if present
|
|
|
|
// (C99 6.7.3p8).
|
|
|
|
const ArrayType *PrettyArrayType = getAsArrayType(Ty);
|
|
|
|
assert(PrettyArrayType && "Not an array type!");
|
2009-09-09 19:08:12 +04:00
|
|
|
|
2008-08-04 11:31:14 +04:00
|
|
|
QualType PtrTy = getPointerType(PrettyArrayType->getElementType());
|
2008-04-02 09:18:44 +04:00
|
|
|
|
|
|
|
// int x[restrict 4] -> int *restrict
|
2009-09-24 23:53:00 +04:00
|
|
|
return getQualifiedType(PtrTy, PrettyArrayType->getIndexTypeQualifiers());
|
2008-04-02 09:18:44 +04:00
|
|
|
}
|
|
|
|
|
2011-01-19 13:06:00 +03:00
|
|
|
QualType ASTContext::getBaseElementType(const ArrayType *array) const {
|
|
|
|
return getBaseElementType(array->getElementType());
|
2009-07-24 03:49:00 +04:00
|
|
|
}
|
|
|
|
|
2011-01-19 13:06:00 +03:00
|
|
|
QualType ASTContext::getBaseElementType(QualType type) const {
|
|
|
|
Qualifiers qs;
|
|
|
|
while (true) {
|
|
|
|
SplitQualType split = type.getSplitDesugaredType();
|
2012-02-08 04:46:36 +04:00
|
|
|
const ArrayType *array = split.Ty->getAsArrayTypeUnsafe();
|
2011-01-19 13:06:00 +03:00
|
|
|
if (!array) break;
|
2009-09-09 19:08:12 +04:00
|
|
|
|
2011-01-19 13:06:00 +03:00
|
|
|
type = array->getElementType();
|
2012-02-08 04:46:36 +04:00
|
|
|
qs.addConsistentQualifiers(split.Quals);
|
2011-01-19 13:06:00 +03:00
|
|
|
}
|
2009-09-09 19:08:12 +04:00
|
|
|
|
2011-01-19 13:06:00 +03:00
|
|
|
return getQualifiedType(type, qs);
|
2008-12-21 06:44:36 +03:00
|
|
|
}
|
|
|
|
|
2009-08-21 20:31:06 +04:00
|
|
|
/// getConstantArrayElementCount - Returns number of constant array elements.
|
2009-09-09 19:08:12 +04:00
|
|
|
uint64_t
|
2009-08-21 20:31:06 +04:00
|
|
|
ASTContext::getConstantArrayElementCount(const ConstantArrayType *CA) const {
|
|
|
|
uint64_t ElementCount = 1;
|
|
|
|
do {
|
|
|
|
ElementCount *= CA->getSize().getZExtValue();
|
|
|
|
CA = dyn_cast<ConstantArrayType>(CA->getElementType());
|
|
|
|
} while (CA);
|
|
|
|
return ElementCount;
|
|
|
|
}
|
|
|
|
|
2007-07-11 21:01:13 +04:00
|
|
|
/// getFloatingRank - Return a relative rank for floating point types.
|
|
|
|
/// This routine will assert if passed a built-in type that isn't a float.
|
2008-04-07 03:38:49 +04:00
|
|
|
static FloatingRank getFloatingRank(QualType T) {
|
2009-09-22 03:43:11 +04:00
|
|
|
if (const ComplexType *CT = T->getAs<ComplexType>())
|
2007-07-11 21:01:13 +04:00
|
|
|
return getFloatingRank(CT->getElementType());
|
2008-04-07 03:38:49 +04:00
|
|
|
|
2009-09-22 03:43:11 +04:00
|
|
|
assert(T->getAs<BuiltinType>() && "getFloatingRank(): not a floating type");
|
|
|
|
switch (T->getAs<BuiltinType>()->getKind()) {
|
2011-09-23 09:06:16 +04:00
|
|
|
default: llvm_unreachable("getFloatingRank(): not a floating type");
|
2011-10-15 03:23:15 +04:00
|
|
|
case BuiltinType::Half: return HalfRank;
|
2007-07-11 21:01:13 +04:00
|
|
|
case BuiltinType::Float: return FloatRank;
|
|
|
|
case BuiltinType::Double: return DoubleRank;
|
|
|
|
case BuiltinType::LongDouble: return LongDoubleRank;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2009-09-09 19:08:12 +04:00
|
|
|
/// getFloatingTypeOfSizeWithinDomain - Returns a real floating
|
|
|
|
/// point or a complex type (based on typeDomain/typeSize).
|
2007-08-27 05:41:48 +04:00
|
|
|
/// 'typeDomain' is a real floating point or complex type.
|
|
|
|
/// 'typeSize' is a real floating point or complex type.
|
2008-04-07 03:58:54 +04:00
|
|
|
QualType ASTContext::getFloatingTypeOfSizeWithinDomain(QualType Size,
|
|
|
|
QualType Domain) const {
|
|
|
|
FloatingRank EltRank = getFloatingRank(Size);
|
|
|
|
if (Domain->isComplexType()) {
|
|
|
|
switch (EltRank) {
|
2012-01-17 06:30:50 +04:00
|
|
|
case HalfRank: llvm_unreachable("Complex half is not supported");
|
2007-08-27 05:27:54 +04:00
|
|
|
case FloatRank: return FloatComplexTy;
|
|
|
|
case DoubleRank: return DoubleComplexTy;
|
|
|
|
case LongDoubleRank: return LongDoubleComplexTy;
|
|
|
|
}
|
|
|
|
}
|
2008-04-07 03:58:54 +04:00
|
|
|
|
|
|
|
assert(Domain->isRealFloatingType() && "Unknown domain!");
|
|
|
|
switch (EltRank) {
|
2012-01-17 06:30:50 +04:00
|
|
|
case HalfRank: llvm_unreachable("Half ranks are not valid here");
|
2008-04-07 03:58:54 +04:00
|
|
|
case FloatRank: return FloatTy;
|
|
|
|
case DoubleRank: return DoubleTy;
|
|
|
|
case LongDoubleRank: return LongDoubleTy;
|
2007-07-11 21:01:13 +04:00
|
|
|
}
|
2012-01-17 06:30:50 +04:00
|
|
|
llvm_unreachable("getFloatingRank(): illegal value for rank");
|
2007-07-11 21:01:13 +04:00
|
|
|
}
|
|
|
|
|
2008-04-07 03:55:33 +04:00
|
|
|
/// getFloatingTypeOrder - Compare the rank of the two specified floating
|
|
|
|
/// point types, ignoring the domain of the type (i.e. 'double' ==
|
|
|
|
/// '_Complex double'). If LHS > RHS, return 1. If LHS == RHS, return 0. If
|
2009-09-09 19:08:12 +04:00
|
|
|
/// LHS < RHS, return -1.
|
2011-01-12 12:06:06 +03:00
|
|
|
int ASTContext::getFloatingTypeOrder(QualType LHS, QualType RHS) const {
|
2008-04-07 03:38:49 +04:00
|
|
|
FloatingRank LHSR = getFloatingRank(LHS);
|
|
|
|
FloatingRank RHSR = getFloatingRank(RHS);
|
2009-09-09 19:08:12 +04:00
|
|
|
|
2008-04-07 03:38:49 +04:00
|
|
|
if (LHSR == RHSR)
|
2007-08-27 19:30:22 +04:00
|
|
|
return 0;
|
2008-04-07 03:38:49 +04:00
|
|
|
if (LHSR > RHSR)
|
2007-08-27 19:30:22 +04:00
|
|
|
return 1;
|
|
|
|
return -1;
|
2007-07-11 21:01:13 +04:00
|
|
|
}
|
|
|
|
|
2008-04-07 02:59:24 +04:00
|
|
|
/// getIntegerRank - Return an integer conversion rank (C99 6.3.1.1p1). This
|
|
|
|
/// routine will assert if passed a built-in type that isn't an integer or enum,
|
|
|
|
/// or if it is not canonicalized.
|
2011-01-19 09:33:43 +03:00
|
|
|
unsigned ASTContext::getIntegerRank(const Type *T) const {
|
2009-10-23 00:10:53 +04:00
|
|
|
assert(T->isCanonicalUnqualified() && "T should be canonicalized");
|
2009-07-14 10:30:34 +04:00
|
|
|
|
2008-04-07 02:59:24 +04:00
|
|
|
switch (cast<BuiltinType>(T)->getKind()) {
|
2011-09-23 09:06:16 +04:00
|
|
|
default: llvm_unreachable("getIntegerRank(): not a built-in integer");
|
2008-04-07 03:55:33 +04:00
|
|
|
case BuiltinType::Bool:
|
2009-02-13 05:31:07 +03:00
|
|
|
return 1 + (getIntWidth(BoolTy) << 3);
|
2008-04-07 03:55:33 +04:00
|
|
|
case BuiltinType::Char_S:
|
|
|
|
case BuiltinType::Char_U:
|
|
|
|
case BuiltinType::SChar:
|
|
|
|
case BuiltinType::UChar:
|
2009-02-13 05:31:07 +03:00
|
|
|
return 2 + (getIntWidth(CharTy) << 3);
|
2008-04-07 03:55:33 +04:00
|
|
|
case BuiltinType::Short:
|
|
|
|
case BuiltinType::UShort:
|
2009-02-13 05:31:07 +03:00
|
|
|
return 3 + (getIntWidth(ShortTy) << 3);
|
2008-04-07 03:55:33 +04:00
|
|
|
case BuiltinType::Int:
|
|
|
|
case BuiltinType::UInt:
|
2009-02-13 05:31:07 +03:00
|
|
|
return 4 + (getIntWidth(IntTy) << 3);
|
2008-04-07 03:55:33 +04:00
|
|
|
case BuiltinType::Long:
|
|
|
|
case BuiltinType::ULong:
|
2009-02-13 05:31:07 +03:00
|
|
|
return 5 + (getIntWidth(LongTy) << 3);
|
2008-04-07 03:55:33 +04:00
|
|
|
case BuiltinType::LongLong:
|
|
|
|
case BuiltinType::ULongLong:
|
2009-02-13 05:31:07 +03:00
|
|
|
return 6 + (getIntWidth(LongLongTy) << 3);
|
2009-04-30 06:43:43 +04:00
|
|
|
case BuiltinType::Int128:
|
|
|
|
case BuiltinType::UInt128:
|
|
|
|
return 7 + (getIntWidth(Int128Ty) << 3);
|
2008-04-07 02:59:24 +04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2009-08-20 08:21:42 +04:00
|
|
|
/// \brief Whether this is a promotable bitfield reference according
|
|
|
|
/// to C99 6.3.1.1p2, bullet 2 (and GCC extensions).
|
|
|
|
///
|
|
|
|
/// \returns the type this bit-field will promote to, or NULL if no
|
|
|
|
/// promotion occurs.
|
2011-01-12 12:06:06 +03:00
|
|
|
QualType ASTContext::isPromotableBitField(Expr *E) const {
|
2010-05-25 00:13:53 +04:00
|
|
|
if (E->isTypeDependent() || E->isValueDependent())
|
|
|
|
return QualType();
|
|
|
|
|
2009-08-20 08:21:42 +04:00
|
|
|
FieldDecl *Field = E->getBitField();
|
|
|
|
if (!Field)
|
|
|
|
return QualType();
|
|
|
|
|
|
|
|
QualType FT = Field->getType();
|
|
|
|
|
2011-10-10 22:28:20 +04:00
|
|
|
uint64_t BitWidth = Field->getBitWidthValue(*this);
|
2009-08-20 08:21:42 +04:00
|
|
|
uint64_t IntSize = getTypeSize(IntTy);
|
|
|
|
// GCC extension compatibility: if the bit-field size is less than or equal
|
|
|
|
// to the size of int, it gets promoted no matter what its type is.
|
|
|
|
// For instance, unsigned long bf : 4 gets promoted to signed int.
|
|
|
|
if (BitWidth < IntSize)
|
|
|
|
return IntTy;
|
|
|
|
|
|
|
|
if (BitWidth == IntSize)
|
|
|
|
return FT->isSignedIntegerType() ? IntTy : UnsignedIntTy;
|
|
|
|
|
|
|
|
// Types bigger than int are not subject to promotions, and therefore act
|
|
|
|
// like the base type.
|
|
|
|
// FIXME: This doesn't quite match what gcc does, but what gcc does here
|
|
|
|
// is ridiculous.
|
|
|
|
return QualType();
|
|
|
|
}
|
|
|
|
|
2009-08-19 11:44:53 +04:00
|
|
|
/// getPromotedIntegerType - Returns the type that Promotable will
|
|
|
|
/// promote to: C99 6.3.1.1p2, assuming that Promotable is a promotable
|
|
|
|
/// integer type.
|
2011-01-12 12:06:06 +03:00
|
|
|
QualType ASTContext::getPromotedIntegerType(QualType Promotable) const {
|
2009-08-19 11:44:53 +04:00
|
|
|
assert(!Promotable.isNull());
|
|
|
|
assert(Promotable->isPromotableIntegerType());
|
2009-12-09 12:09:27 +03:00
|
|
|
if (const EnumType *ET = Promotable->getAs<EnumType>())
|
|
|
|
return ET->getDecl()->getPromotionType();
|
2011-10-26 11:22:48 +04:00
|
|
|
|
|
|
|
if (const BuiltinType *BT = Promotable->getAs<BuiltinType>()) {
|
|
|
|
// C++ [conv.prom]: A prvalue of type char16_t, char32_t, or wchar_t
|
|
|
|
// (3.9.1) can be converted to a prvalue of the first of the following
|
|
|
|
// types that can represent all the values of its underlying type:
|
|
|
|
// int, unsigned int, long int, unsigned long int, long long int, or
|
|
|
|
// unsigned long long int [...]
|
|
|
|
// FIXME: Is there some better way to compute this?
|
|
|
|
if (BT->getKind() == BuiltinType::WChar_S ||
|
|
|
|
BT->getKind() == BuiltinType::WChar_U ||
|
|
|
|
BT->getKind() == BuiltinType::Char16 ||
|
|
|
|
BT->getKind() == BuiltinType::Char32) {
|
|
|
|
bool FromIsSigned = BT->getKind() == BuiltinType::WChar_S;
|
|
|
|
uint64_t FromSize = getTypeSize(BT);
|
|
|
|
QualType PromoteTypes[] = { IntTy, UnsignedIntTy, LongTy, UnsignedLongTy,
|
|
|
|
LongLongTy, UnsignedLongLongTy };
|
|
|
|
for (size_t Idx = 0; Idx < llvm::array_lengthof(PromoteTypes); ++Idx) {
|
|
|
|
uint64_t ToSize = getTypeSize(PromoteTypes[Idx]);
|
|
|
|
if (FromSize < ToSize ||
|
|
|
|
(FromSize == ToSize &&
|
|
|
|
FromIsSigned == PromoteTypes[Idx]->isSignedIntegerType()))
|
|
|
|
return PromoteTypes[Idx];
|
|
|
|
}
|
|
|
|
llvm_unreachable("char type should fit into long long");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// At this point, we should have a signed or unsigned integer type.
|
2009-08-19 11:44:53 +04:00
|
|
|
if (Promotable->isSignedIntegerType())
|
|
|
|
return IntTy;
|
|
|
|
uint64_t PromotableSize = getTypeSize(Promotable);
|
|
|
|
uint64_t IntSize = getTypeSize(IntTy);
|
|
|
|
assert(Promotable->isUnsignedIntegerType() && PromotableSize <= IntSize);
|
|
|
|
return (PromotableSize != IntSize) ? IntTy : UnsignedIntTy;
|
|
|
|
}
|
|
|
|
|
2011-07-02 02:22:50 +04:00
|
|
|
/// \brief Recurses in pointer/array types until it finds an objc retainable
|
|
|
|
/// type and returns its ownership.
|
|
|
|
Qualifiers::ObjCLifetime ASTContext::getInnerObjCOwnership(QualType T) const {
|
|
|
|
while (!T.isNull()) {
|
|
|
|
if (T.getObjCLifetime() != Qualifiers::OCL_None)
|
|
|
|
return T.getObjCLifetime();
|
|
|
|
if (T->isArrayType())
|
|
|
|
T = getBaseElementType(T);
|
|
|
|
else if (const PointerType *PT = T->getAs<PointerType>())
|
|
|
|
T = PT->getPointeeType();
|
|
|
|
else if (const ReferenceType *RT = T->getAs<ReferenceType>())
|
2011-07-02 03:01:46 +04:00
|
|
|
T = RT->getPointeeType();
|
2011-07-02 02:22:50 +04:00
|
|
|
else
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
return Qualifiers::OCL_None;
|
|
|
|
}
|
|
|
|
|
2009-09-09 19:08:12 +04:00
|
|
|
/// getIntegerTypeOrder - Returns the highest ranked integer type:
|
2008-04-07 03:55:33 +04:00
|
|
|
/// C99 6.3.1.8p1. If LHS > RHS, return 1. If LHS == RHS, return 0. If
|
2009-09-09 19:08:12 +04:00
|
|
|
/// LHS < RHS, return -1.
|
2011-01-12 12:06:06 +03:00
|
|
|
int ASTContext::getIntegerTypeOrder(QualType LHS, QualType RHS) const {
|
2011-01-19 09:33:43 +03:00
|
|
|
const Type *LHSC = getCanonicalType(LHS).getTypePtr();
|
|
|
|
const Type *RHSC = getCanonicalType(RHS).getTypePtr();
|
2008-04-07 03:55:33 +04:00
|
|
|
if (LHSC == RHSC) return 0;
|
2009-09-09 19:08:12 +04:00
|
|
|
|
2008-04-07 02:59:24 +04:00
|
|
|
bool LHSUnsigned = LHSC->isUnsignedIntegerType();
|
|
|
|
bool RHSUnsigned = RHSC->isUnsignedIntegerType();
|
2009-09-09 19:08:12 +04:00
|
|
|
|
2008-04-07 03:55:33 +04:00
|
|
|
unsigned LHSRank = getIntegerRank(LHSC);
|
|
|
|
unsigned RHSRank = getIntegerRank(RHSC);
|
2009-09-09 19:08:12 +04:00
|
|
|
|
2008-04-07 03:55:33 +04:00
|
|
|
if (LHSUnsigned == RHSUnsigned) { // Both signed or both unsigned.
|
|
|
|
if (LHSRank == RHSRank) return 0;
|
|
|
|
return LHSRank > RHSRank ? 1 : -1;
|
|
|
|
}
|
2009-09-09 19:08:12 +04:00
|
|
|
|
2008-04-07 03:55:33 +04:00
|
|
|
// Otherwise, the LHS is signed and the RHS is unsigned or visa versa.
|
|
|
|
if (LHSUnsigned) {
|
|
|
|
// If the unsigned [LHS] type is larger, return it.
|
|
|
|
if (LHSRank >= RHSRank)
|
|
|
|
return 1;
|
2009-09-09 19:08:12 +04:00
|
|
|
|
2008-04-07 03:55:33 +04:00
|
|
|
// If the signed type can represent all values of the unsigned type, it
|
|
|
|
// wins. Because we are dealing with 2's complement and types that are
|
2009-09-09 19:08:12 +04:00
|
|
|
// powers of two larger than each other, this is always safe.
|
2008-04-07 03:55:33 +04:00
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
// If the unsigned [RHS] type is larger, return it.
|
|
|
|
if (RHSRank >= LHSRank)
|
|
|
|
return -1;
|
2009-09-09 19:08:12 +04:00
|
|
|
|
2008-04-07 03:55:33 +04:00
|
|
|
// If the signed type can represent all values of the unsigned type, it
|
|
|
|
// wins. Because we are dealing with 2's complement and types that are
|
2009-09-09 19:08:12 +04:00
|
|
|
// powers of two larger than each other, this is always safe.
|
2008-04-07 03:55:33 +04:00
|
|
|
return 1;
|
2007-07-11 21:01:13 +04:00
|
|
|
}
|
2007-08-17 09:31:46 +04:00
|
|
|
|
2009-11-15 00:45:58 +03:00
|
|
|
static RecordDecl *
|
2011-03-09 17:09:51 +03:00
|
|
|
CreateRecordDecl(const ASTContext &Ctx, RecordDecl::TagKind TK,
|
|
|
|
DeclContext *DC, IdentifierInfo *Id) {
|
|
|
|
SourceLocation Loc;
|
2012-03-11 11:00:24 +04:00
|
|
|
if (Ctx.getLangOpts().CPlusPlus)
|
2011-03-09 17:09:51 +03:00
|
|
|
return CXXRecordDecl::Create(Ctx, TK, DC, Loc, Loc, Id);
|
2009-11-15 00:45:58 +03:00
|
|
|
else
|
2011-03-09 17:09:51 +03:00
|
|
|
return RecordDecl::Create(Ctx, TK, DC, Loc, Loc, Id);
|
2009-11-15 00:45:58 +03:00
|
|
|
}
|
2011-03-09 17:09:51 +03:00
|
|
|
|
2009-09-09 19:08:12 +04:00
|
|
|
// getCFConstantStringType - Return the type used for constant CFStrings.
|
2011-01-12 12:06:06 +03:00
|
|
|
QualType ASTContext::getCFConstantStringType() const {
|
2007-08-17 09:31:46 +04:00
|
|
|
if (!CFConstantStringTypeDecl) {
|
2009-09-09 19:08:12 +04:00
|
|
|
CFConstantStringTypeDecl =
|
2011-03-09 17:09:51 +03:00
|
|
|
CreateRecordDecl(*this, TTK_Struct, TUDecl,
|
2009-11-15 00:45:58 +03:00
|
|
|
&Idents.get("NSConstantString"));
|
2010-02-05 04:33:36 +03:00
|
|
|
CFConstantStringTypeDecl->startDefinition();
|
2009-11-15 00:45:58 +03:00
|
|
|
|
2007-11-19 03:25:30 +03:00
|
|
|
QualType FieldTypes[4];
|
2009-09-09 19:08:12 +04:00
|
|
|
|
2007-08-17 09:31:46 +04:00
|
|
|
// const int *isa;
|
2009-09-24 23:53:00 +04:00
|
|
|
FieldTypes[0] = getPointerType(IntTy.withConst());
|
2007-11-19 03:25:30 +03:00
|
|
|
// int flags;
|
|
|
|
FieldTypes[1] = IntTy;
|
2007-08-17 09:31:46 +04:00
|
|
|
// const char *str;
|
2009-09-24 23:53:00 +04:00
|
|
|
FieldTypes[2] = getPointerType(CharTy.withConst());
|
2007-08-17 09:31:46 +04:00
|
|
|
// long length;
|
2009-09-09 19:08:12 +04:00
|
|
|
FieldTypes[3] = LongTy;
|
|
|
|
|
2008-12-11 19:49:14 +03:00
|
|
|
// Create fields
|
|
|
|
for (unsigned i = 0; i < 4; ++i) {
|
2009-09-09 19:08:12 +04:00
|
|
|
FieldDecl *Field = FieldDecl::Create(*this, CFConstantStringTypeDecl,
|
2011-03-08 11:55:46 +03:00
|
|
|
SourceLocation(),
|
2008-12-11 19:49:14 +03:00
|
|
|
SourceLocation(), 0,
|
2009-12-07 05:54:59 +03:00
|
|
|
FieldTypes[i], /*TInfo=*/0,
|
2009-09-09 19:08:12 +04:00
|
|
|
/*BitWidth=*/0,
|
2011-06-11 21:19:42 +04:00
|
|
|
/*Mutable=*/false,
|
|
|
|
/*HasInit=*/false);
|
2010-05-01 01:35:41 +04:00
|
|
|
Field->setAccess(AS_public);
|
2009-06-30 06:36:12 +04:00
|
|
|
CFConstantStringTypeDecl->addDecl(Field);
|
2008-12-11 19:49:14 +03:00
|
|
|
}
|
|
|
|
|
2010-02-11 04:19:42 +03:00
|
|
|
CFConstantStringTypeDecl->completeDefinition();
|
2007-08-17 09:31:46 +04:00
|
|
|
}
|
2009-09-09 19:08:12 +04:00
|
|
|
|
2007-08-17 09:31:46 +04:00
|
|
|
return getTagDeclType(CFConstantStringTypeDecl);
|
2007-09-11 19:32:40 +04:00
|
|
|
}
|
2007-10-11 05:00:40 +04:00
|
|
|
|
2009-04-24 02:29:11 +04:00
|
|
|
void ASTContext::setCFConstantStringType(QualType T) {
|
2009-07-30 01:53:49 +04:00
|
|
|
const RecordType *Rec = T->getAs<RecordType>();
|
2009-04-24 02:29:11 +04:00
|
|
|
assert(Rec && "Invalid CFConstantStringType");
|
|
|
|
CFConstantStringTypeDecl = Rec->getDecl();
|
|
|
|
}
|
|
|
|
|
2011-01-12 12:06:06 +03:00
|
|
|
QualType ASTContext::getBlockDescriptorType() const {
|
2009-10-20 06:12:22 +04:00
|
|
|
if (BlockDescriptorType)
|
|
|
|
return getTagDeclType(BlockDescriptorType);
|
|
|
|
|
|
|
|
RecordDecl *T;
|
|
|
|
// FIXME: Needs the FlagAppleBlock bit.
|
2011-03-09 17:09:51 +03:00
|
|
|
T = CreateRecordDecl(*this, TTK_Struct, TUDecl,
|
2009-11-15 00:45:58 +03:00
|
|
|
&Idents.get("__block_descriptor"));
|
2010-02-05 04:33:36 +03:00
|
|
|
T->startDefinition();
|
2009-10-20 06:12:22 +04:00
|
|
|
|
|
|
|
QualType FieldTypes[] = {
|
|
|
|
UnsignedLongTy,
|
|
|
|
UnsignedLongTy,
|
|
|
|
};
|
|
|
|
|
|
|
|
const char *FieldNames[] = {
|
|
|
|
"reserved",
|
2009-10-22 04:49:09 +04:00
|
|
|
"Size"
|
2009-10-20 06:12:22 +04:00
|
|
|
};
|
|
|
|
|
|
|
|
for (size_t i = 0; i < 2; ++i) {
|
2011-03-08 11:55:46 +03:00
|
|
|
FieldDecl *Field = FieldDecl::Create(*this, T, SourceLocation(),
|
2009-10-20 06:12:22 +04:00
|
|
|
SourceLocation(),
|
|
|
|
&Idents.get(FieldNames[i]),
|
2009-12-07 05:54:59 +03:00
|
|
|
FieldTypes[i], /*TInfo=*/0,
|
2009-10-20 06:12:22 +04:00
|
|
|
/*BitWidth=*/0,
|
2011-06-11 21:19:42 +04:00
|
|
|
/*Mutable=*/false,
|
|
|
|
/*HasInit=*/false);
|
2010-05-01 01:35:41 +04:00
|
|
|
Field->setAccess(AS_public);
|
2009-10-20 06:12:22 +04:00
|
|
|
T->addDecl(Field);
|
|
|
|
}
|
|
|
|
|
2010-02-11 04:19:42 +03:00
|
|
|
T->completeDefinition();
|
2009-10-20 06:12:22 +04:00
|
|
|
|
|
|
|
BlockDescriptorType = T;
|
|
|
|
|
|
|
|
return getTagDeclType(BlockDescriptorType);
|
|
|
|
}
|
|
|
|
|
2011-01-12 12:06:06 +03:00
|
|
|
QualType ASTContext::getBlockDescriptorExtendedType() const {
|
2009-10-22 04:49:09 +04:00
|
|
|
if (BlockDescriptorExtendedType)
|
|
|
|
return getTagDeclType(BlockDescriptorExtendedType);
|
|
|
|
|
|
|
|
RecordDecl *T;
|
|
|
|
// FIXME: Needs the FlagAppleBlock bit.
|
2011-03-09 17:09:51 +03:00
|
|
|
T = CreateRecordDecl(*this, TTK_Struct, TUDecl,
|
2009-11-15 00:45:58 +03:00
|
|
|
&Idents.get("__block_descriptor_withcopydispose"));
|
2010-02-05 04:33:36 +03:00
|
|
|
T->startDefinition();
|
2009-10-22 04:49:09 +04:00
|
|
|
|
|
|
|
QualType FieldTypes[] = {
|
|
|
|
UnsignedLongTy,
|
|
|
|
UnsignedLongTy,
|
|
|
|
getPointerType(VoidPtrTy),
|
|
|
|
getPointerType(VoidPtrTy)
|
|
|
|
};
|
|
|
|
|
|
|
|
const char *FieldNames[] = {
|
|
|
|
"reserved",
|
|
|
|
"Size",
|
|
|
|
"CopyFuncPtr",
|
|
|
|
"DestroyFuncPtr"
|
|
|
|
};
|
|
|
|
|
|
|
|
for (size_t i = 0; i < 4; ++i) {
|
2011-03-08 11:55:46 +03:00
|
|
|
FieldDecl *Field = FieldDecl::Create(*this, T, SourceLocation(),
|
2009-10-22 04:49:09 +04:00
|
|
|
SourceLocation(),
|
|
|
|
&Idents.get(FieldNames[i]),
|
2009-12-07 05:54:59 +03:00
|
|
|
FieldTypes[i], /*TInfo=*/0,
|
2009-10-22 04:49:09 +04:00
|
|
|
/*BitWidth=*/0,
|
2011-06-11 21:19:42 +04:00
|
|
|
/*Mutable=*/false,
|
|
|
|
/*HasInit=*/false);
|
2010-05-01 01:35:41 +04:00
|
|
|
Field->setAccess(AS_public);
|
2009-10-22 04:49:09 +04:00
|
|
|
T->addDecl(Field);
|
|
|
|
}
|
|
|
|
|
2010-02-11 04:19:42 +03:00
|
|
|
T->completeDefinition();
|
2009-10-22 04:49:09 +04:00
|
|
|
|
|
|
|
BlockDescriptorExtendedType = T;
|
|
|
|
|
|
|
|
return getTagDeclType(BlockDescriptorExtendedType);
|
|
|
|
}
|
|
|
|
|
2011-01-12 12:06:06 +03:00
|
|
|
bool ASTContext::BlockRequiresCopying(QualType Ty) const {
|
2011-06-16 03:02:42 +04:00
|
|
|
if (Ty->isObjCRetainableType())
|
2009-10-21 22:16:27 +04:00
|
|
|
return true;
|
2012-03-11 11:00:24 +04:00
|
|
|
if (getLangOpts().CPlusPlus) {
|
2010-11-17 03:21:28 +03:00
|
|
|
if (const RecordType *RT = Ty->getAs<RecordType>()) {
|
|
|
|
CXXRecordDecl *RD = cast<CXXRecordDecl>(RT->getDecl());
|
2011-05-26 00:50:04 +04:00
|
|
|
return RD->hasConstCopyConstructor();
|
2010-11-17 03:21:28 +03:00
|
|
|
|
|
|
|
}
|
|
|
|
}
|
2009-10-21 22:16:27 +04:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2011-01-12 12:06:06 +03:00
|
|
|
QualType
|
2011-07-23 14:55:15 +04:00
|
|
|
ASTContext::BuildByRefType(StringRef DeclName, QualType Ty) const {
|
2009-10-21 22:16:27 +04:00
|
|
|
// type = struct __Block_byref_1_X {
|
2009-10-21 07:49:08 +04:00
|
|
|
// void *__isa;
|
2009-10-21 22:16:27 +04:00
|
|
|
// struct __Block_byref_1_X *__forwarding;
|
2009-10-21 07:49:08 +04:00
|
|
|
// unsigned int __flags;
|
|
|
|
// unsigned int __size;
|
2010-08-22 05:00:03 +04:00
|
|
|
// void *__copy_helper; // as needed
|
|
|
|
// void *__destroy_help // as needed
|
2009-10-21 22:16:27 +04:00
|
|
|
// int X;
|
2009-10-21 07:49:08 +04:00
|
|
|
// } *
|
|
|
|
|
2009-10-21 22:16:27 +04:00
|
|
|
bool HasCopyAndDispose = BlockRequiresCopying(Ty);
|
|
|
|
|
|
|
|
// FIXME: Move up
|
2012-02-05 06:13:05 +04:00
|
|
|
SmallString<36> Name;
|
2009-10-24 13:57:09 +04:00
|
|
|
llvm::raw_svector_ostream(Name) << "__Block_byref_" <<
|
|
|
|
++UniqueBlockByRefTypeID << '_' << DeclName;
|
2009-10-21 22:16:27 +04:00
|
|
|
RecordDecl *T;
|
2011-03-09 17:09:51 +03:00
|
|
|
T = CreateRecordDecl(*this, TTK_Struct, TUDecl, &Idents.get(Name.str()));
|
2009-10-21 22:16:27 +04:00
|
|
|
T->startDefinition();
|
|
|
|
QualType Int32Ty = IntTy;
|
|
|
|
assert(getIntWidth(IntTy) == 32 && "non-32bit int not supported");
|
|
|
|
QualType FieldTypes[] = {
|
|
|
|
getPointerType(VoidPtrTy),
|
|
|
|
getPointerType(getTagDeclType(T)),
|
|
|
|
Int32Ty,
|
|
|
|
Int32Ty,
|
|
|
|
getPointerType(VoidPtrTy),
|
|
|
|
getPointerType(VoidPtrTy),
|
|
|
|
Ty
|
|
|
|
};
|
|
|
|
|
2011-07-23 14:55:15 +04:00
|
|
|
StringRef FieldNames[] = {
|
2009-10-21 22:16:27 +04:00
|
|
|
"__isa",
|
|
|
|
"__forwarding",
|
|
|
|
"__flags",
|
|
|
|
"__size",
|
|
|
|
"__copy_helper",
|
|
|
|
"__destroy_helper",
|
|
|
|
DeclName,
|
|
|
|
};
|
|
|
|
|
|
|
|
for (size_t i = 0; i < 7; ++i) {
|
|
|
|
if (!HasCopyAndDispose && i >=4 && i <= 5)
|
|
|
|
continue;
|
|
|
|
FieldDecl *Field = FieldDecl::Create(*this, T, SourceLocation(),
|
2011-03-08 11:55:46 +03:00
|
|
|
SourceLocation(),
|
2009-10-21 22:16:27 +04:00
|
|
|
&Idents.get(FieldNames[i]),
|
2009-12-07 05:54:59 +03:00
|
|
|
FieldTypes[i], /*TInfo=*/0,
|
2011-06-11 21:19:42 +04:00
|
|
|
/*BitWidth=*/0, /*Mutable=*/false,
|
|
|
|
/*HasInit=*/false);
|
2010-05-01 01:35:41 +04:00
|
|
|
Field->setAccess(AS_public);
|
2009-10-21 22:16:27 +04:00
|
|
|
T->addDecl(Field);
|
|
|
|
}
|
|
|
|
|
2010-02-11 04:19:42 +03:00
|
|
|
T->completeDefinition();
|
2009-10-21 22:16:27 +04:00
|
|
|
|
|
|
|
return getPointerType(getTagDeclType(T));
|
2009-10-21 07:49:08 +04:00
|
|
|
}
|
|
|
|
|
2011-09-08 05:46:34 +04:00
|
|
|
TypedefDecl *ASTContext::getObjCInstanceTypeDecl() {
|
|
|
|
if (!ObjCInstanceTypeDecl)
|
|
|
|
ObjCInstanceTypeDecl = TypedefDecl::Create(*this,
|
|
|
|
getTranslationUnitDecl(),
|
|
|
|
SourceLocation(),
|
|
|
|
SourceLocation(),
|
|
|
|
&Idents.get("instancetype"),
|
|
|
|
getTrivialTypeSourceInfo(getObjCIdType()));
|
|
|
|
return ObjCInstanceTypeDecl;
|
|
|
|
}
|
|
|
|
|
2007-10-29 09:33:42 +03:00
|
|
|
// This returns true if a type has been typedefed to BOOL:
|
|
|
|
// typedef <type> BOOL;
|
2007-10-30 23:27:44 +03:00
|
|
|
static bool isTypeTypedefedAsBOOL(QualType T) {
|
2007-10-29 09:33:42 +03:00
|
|
|
if (const TypedefType *TT = dyn_cast<TypedefType>(T))
|
2008-11-24 06:52:59 +03:00
|
|
|
if (IdentifierInfo *II = TT->getDecl()->getIdentifier())
|
|
|
|
return II->isStr("BOOL");
|
2009-09-09 19:08:12 +04:00
|
|
|
|
2007-10-29 08:01:08 +03:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2008-01-07 22:49:32 +03:00
|
|
|
/// getObjCEncodingTypeSize returns size of type for objective-c encoding
|
2007-10-30 01:57:28 +03:00
|
|
|
/// purpose.
|
2011-01-12 12:06:06 +03:00
|
|
|
CharUnits ASTContext::getObjCEncodingTypeSize(QualType type) const {
|
2011-05-27 05:19:52 +04:00
|
|
|
if (!type->isIncompleteArrayType() && type->isIncompleteType())
|
|
|
|
return CharUnits::Zero();
|
|
|
|
|
2010-01-11 20:06:35 +03:00
|
|
|
CharUnits sz = getTypeSizeInChars(type);
|
2009-09-09 19:08:12 +04:00
|
|
|
|
2007-10-30 01:57:28 +03:00
|
|
|
// Make all integer and enum types at least as large as an int
|
2010-06-16 04:17:44 +04:00
|
|
|
if (sz.isPositive() && type->isIntegralOrEnumerationType())
|
2010-01-11 20:06:35 +03:00
|
|
|
sz = std::max(sz, getTypeSizeInChars(IntTy));
|
2007-10-30 01:57:28 +03:00
|
|
|
// Treat arrays as pointers, since that's how they're passed in.
|
|
|
|
else if (type->isArrayType())
|
2010-01-11 20:06:35 +03:00
|
|
|
sz = getTypeSizeInChars(VoidPtrTy);
|
2010-01-11 22:19:56 +03:00
|
|
|
return sz;
|
2010-01-11 20:06:35 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
static inline
|
|
|
|
std::string charUnitsToString(const CharUnits &CU) {
|
|
|
|
return llvm::itostr(CU.getQuantity());
|
2007-10-30 01:57:28 +03:00
|
|
|
}
|
|
|
|
|
2011-02-07 13:33:21 +03:00
|
|
|
/// getObjCEncodingForBlock - Return the encoded type for this block
|
2009-11-17 22:33:30 +03:00
|
|
|
/// declaration.
|
2011-02-07 13:33:21 +03:00
|
|
|
std::string ASTContext::getObjCEncodingForBlock(const BlockExpr *Expr) const {
|
|
|
|
std::string S;
|
|
|
|
|
2009-11-17 22:33:30 +03:00
|
|
|
const BlockDecl *Decl = Expr->getBlockDecl();
|
|
|
|
QualType BlockTy =
|
|
|
|
Expr->getType()->getAs<BlockPointerType>()->getPointeeType();
|
|
|
|
// Encode result type.
|
2010-06-04 23:02:56 +04:00
|
|
|
getObjCEncodingForType(BlockTy->getAs<FunctionType>()->getResultType(), S);
|
2009-11-17 22:33:30 +03:00
|
|
|
// Compute size of all parameters.
|
|
|
|
// Start with computing size of a pointer in number of bytes.
|
|
|
|
// FIXME: There might(should) be a better way of doing this computation!
|
|
|
|
SourceLocation Loc;
|
2010-01-11 20:06:35 +03:00
|
|
|
CharUnits PtrSize = getTypeSizeInChars(VoidPtrTy);
|
|
|
|
CharUnits ParmOffset = PtrSize;
|
2010-04-08 22:06:22 +04:00
|
|
|
for (BlockDecl::param_const_iterator PI = Decl->param_begin(),
|
2009-11-17 22:33:30 +03:00
|
|
|
E = Decl->param_end(); PI != E; ++PI) {
|
|
|
|
QualType PType = (*PI)->getType();
|
2010-01-11 22:19:56 +03:00
|
|
|
CharUnits sz = getObjCEncodingTypeSize(PType);
|
2010-01-11 20:06:35 +03:00
|
|
|
assert (sz.isPositive() && "BlockExpr - Incomplete param type");
|
2009-11-17 22:33:30 +03:00
|
|
|
ParmOffset += sz;
|
|
|
|
}
|
|
|
|
// Size of the argument frame
|
2010-01-11 20:06:35 +03:00
|
|
|
S += charUnitsToString(ParmOffset);
|
2009-11-17 22:33:30 +03:00
|
|
|
// Block pointer and offset.
|
|
|
|
S += "@?0";
|
|
|
|
|
|
|
|
// Argument types.
|
|
|
|
ParmOffset = PtrSize;
|
|
|
|
for (BlockDecl::param_const_iterator PI = Decl->param_begin(), E =
|
|
|
|
Decl->param_end(); PI != E; ++PI) {
|
|
|
|
ParmVarDecl *PVDecl = *PI;
|
|
|
|
QualType PType = PVDecl->getOriginalType();
|
|
|
|
if (const ArrayType *AT =
|
|
|
|
dyn_cast<ArrayType>(PType->getCanonicalTypeInternal())) {
|
|
|
|
// Use array's original type only if it has known number of
|
2010-12-30 17:05:53 +03:00
|
|
|
// elements.
|
|
|
|
if (!isa<ConstantArrayType>(AT))
|
|
|
|
PType = PVDecl->getType();
|
|
|
|
} else if (PType->isFunctionType())
|
|
|
|
PType = PVDecl->getType();
|
|
|
|
getObjCEncodingForType(PType, S);
|
|
|
|
S += charUnitsToString(ParmOffset);
|
|
|
|
ParmOffset += getObjCEncodingTypeSize(PType);
|
|
|
|
}
|
2011-02-07 13:33:21 +03:00
|
|
|
|
|
|
|
return S;
|
2010-12-30 17:05:53 +03:00
|
|
|
}
|
|
|
|
|
2011-05-27 05:19:52 +04:00
|
|
|
bool ASTContext::getObjCEncodingForFunctionDecl(const FunctionDecl *Decl,
|
2010-12-30 17:05:53 +03:00
|
|
|
std::string& S) {
|
|
|
|
// Encode result type.
|
|
|
|
getObjCEncodingForType(Decl->getResultType(), S);
|
|
|
|
CharUnits ParmOffset;
|
|
|
|
// Compute size of all parameters.
|
|
|
|
for (FunctionDecl::param_const_iterator PI = Decl->param_begin(),
|
|
|
|
E = Decl->param_end(); PI != E; ++PI) {
|
|
|
|
QualType PType = (*PI)->getType();
|
|
|
|
CharUnits sz = getObjCEncodingTypeSize(PType);
|
2011-05-27 05:19:52 +04:00
|
|
|
if (sz.isZero())
|
|
|
|
return true;
|
|
|
|
|
2010-12-30 17:05:53 +03:00
|
|
|
assert (sz.isPositive() &&
|
2011-05-27 05:19:52 +04:00
|
|
|
"getObjCEncodingForFunctionDecl - Incomplete param type");
|
2010-12-30 17:05:53 +03:00
|
|
|
ParmOffset += sz;
|
|
|
|
}
|
|
|
|
S += charUnitsToString(ParmOffset);
|
|
|
|
ParmOffset = CharUnits::Zero();
|
|
|
|
|
|
|
|
// Argument types.
|
|
|
|
for (FunctionDecl::param_const_iterator PI = Decl->param_begin(),
|
|
|
|
E = Decl->param_end(); PI != E; ++PI) {
|
|
|
|
ParmVarDecl *PVDecl = *PI;
|
|
|
|
QualType PType = PVDecl->getOriginalType();
|
|
|
|
if (const ArrayType *AT =
|
|
|
|
dyn_cast<ArrayType>(PType->getCanonicalTypeInternal())) {
|
|
|
|
// Use array's original type only if it has known number of
|
2009-11-17 22:33:30 +03:00
|
|
|
// elements.
|
|
|
|
if (!isa<ConstantArrayType>(AT))
|
|
|
|
PType = PVDecl->getType();
|
|
|
|
} else if (PType->isFunctionType())
|
|
|
|
PType = PVDecl->getType();
|
|
|
|
getObjCEncodingForType(PType, S);
|
2010-01-11 20:06:35 +03:00
|
|
|
S += charUnitsToString(ParmOffset);
|
2010-01-11 22:19:56 +03:00
|
|
|
ParmOffset += getObjCEncodingTypeSize(PType);
|
2009-11-17 22:33:30 +03:00
|
|
|
}
|
2011-05-27 05:19:52 +04:00
|
|
|
|
|
|
|
return false;
|
2009-11-17 22:33:30 +03:00
|
|
|
}
|
|
|
|
|
2011-11-30 05:57:58 +04:00
|
|
|
/// getObjCEncodingForMethodParameter - Return the encoded type for a single
|
|
|
|
/// method parameter or return type. If Extended, include class names and
|
|
|
|
/// block object types.
|
|
|
|
void ASTContext::getObjCEncodingForMethodParameter(Decl::ObjCDeclQualifier QT,
|
|
|
|
QualType T, std::string& S,
|
|
|
|
bool Extended) const {
|
|
|
|
// Encode type qualifer, 'in', 'inout', etc. for the parameter.
|
|
|
|
getObjCEncodingForTypeQualifier(QT, S);
|
|
|
|
// Encode parameter type.
|
|
|
|
getObjCEncodingForTypeImpl(T, S, true, true, 0,
|
|
|
|
true /*OutermostType*/,
|
|
|
|
false /*EncodingProperty*/,
|
|
|
|
false /*StructField*/,
|
|
|
|
Extended /*EncodeBlockParameters*/,
|
|
|
|
Extended /*EncodeClassNames*/);
|
|
|
|
}
|
|
|
|
|
2008-01-07 22:49:32 +03:00
|
|
|
/// getObjCEncodingForMethodDecl - Return the encoded type for this method
|
2007-10-30 01:57:28 +03:00
|
|
|
/// declaration.
|
2011-05-27 05:19:52 +04:00
|
|
|
bool ASTContext::getObjCEncodingForMethodDecl(const ObjCMethodDecl *Decl,
|
2011-11-30 05:57:58 +04:00
|
|
|
std::string& S,
|
|
|
|
bool Extended) const {
|
2008-08-28 08:38:10 +04:00
|
|
|
// FIXME: This is not very efficient.
|
2011-11-30 05:57:58 +04:00
|
|
|
// Encode return type.
|
|
|
|
getObjCEncodingForMethodParameter(Decl->getObjCDeclQualifier(),
|
|
|
|
Decl->getResultType(), S, Extended);
|
2007-10-30 01:57:28 +03:00
|
|
|
// Compute size of all parameters.
|
|
|
|
// Start with computing size of a pointer in number of bytes.
|
|
|
|
// FIXME: There might(should) be a better way of doing this computation!
|
|
|
|
SourceLocation Loc;
|
2010-01-11 20:06:35 +03:00
|
|
|
CharUnits PtrSize = getTypeSizeInChars(VoidPtrTy);
|
2007-10-30 01:57:28 +03:00
|
|
|
// The first two arguments (self and _cmd) are pointers; account for
|
|
|
|
// their size.
|
2010-01-11 20:06:35 +03:00
|
|
|
CharUnits ParmOffset = 2 * PtrSize;
|
2011-10-03 10:37:04 +04:00
|
|
|
for (ObjCMethodDecl::param_const_iterator PI = Decl->param_begin(),
|
2010-04-09 01:29:11 +04:00
|
|
|
E = Decl->sel_param_end(); PI != E; ++PI) {
|
2009-02-20 21:43:26 +03:00
|
|
|
QualType PType = (*PI)->getType();
|
2010-01-11 22:19:56 +03:00
|
|
|
CharUnits sz = getObjCEncodingTypeSize(PType);
|
2011-05-27 05:19:52 +04:00
|
|
|
if (sz.isZero())
|
|
|
|
return true;
|
|
|
|
|
2010-01-11 20:06:35 +03:00
|
|
|
assert (sz.isPositive() &&
|
|
|
|
"getObjCEncodingForMethodDecl - Incomplete param type");
|
2007-10-30 01:57:28 +03:00
|
|
|
ParmOffset += sz;
|
|
|
|
}
|
2010-01-11 20:06:35 +03:00
|
|
|
S += charUnitsToString(ParmOffset);
|
2007-10-30 01:57:28 +03:00
|
|
|
S += "@0:";
|
2010-01-11 20:06:35 +03:00
|
|
|
S += charUnitsToString(PtrSize);
|
2009-09-09 19:08:12 +04:00
|
|
|
|
2007-10-30 01:57:28 +03:00
|
|
|
// Argument types.
|
|
|
|
ParmOffset = 2 * PtrSize;
|
2011-10-03 10:37:04 +04:00
|
|
|
for (ObjCMethodDecl::param_const_iterator PI = Decl->param_begin(),
|
2010-04-09 01:29:11 +04:00
|
|
|
E = Decl->sel_param_end(); PI != E; ++PI) {
|
2011-10-03 10:37:04 +04:00
|
|
|
const ParmVarDecl *PVDecl = *PI;
|
2009-09-09 19:08:12 +04:00
|
|
|
QualType PType = PVDecl->getOriginalType();
|
2008-12-21 02:29:59 +03:00
|
|
|
if (const ArrayType *AT =
|
2009-04-14 04:03:58 +04:00
|
|
|
dyn_cast<ArrayType>(PType->getCanonicalTypeInternal())) {
|
|
|
|
// Use array's original type only if it has known number of
|
|
|
|
// elements.
|
2009-04-14 04:40:09 +04:00
|
|
|
if (!isa<ConstantArrayType>(AT))
|
2009-04-14 04:03:58 +04:00
|
|
|
PType = PVDecl->getType();
|
|
|
|
} else if (PType->isFunctionType())
|
|
|
|
PType = PVDecl->getType();
|
2011-11-30 05:57:58 +04:00
|
|
|
getObjCEncodingForMethodParameter(PVDecl->getObjCDeclQualifier(),
|
|
|
|
PType, S, Extended);
|
2010-01-11 20:06:35 +03:00
|
|
|
S += charUnitsToString(ParmOffset);
|
2010-01-11 22:19:56 +03:00
|
|
|
ParmOffset += getObjCEncodingTypeSize(PType);
|
2007-10-30 01:57:28 +03:00
|
|
|
}
|
2011-05-27 05:19:52 +04:00
|
|
|
|
|
|
|
return false;
|
2007-10-30 01:57:28 +03:00
|
|
|
}
|
|
|
|
|
2008-08-28 08:38:10 +04:00
|
|
|
/// getObjCEncodingForPropertyDecl - Return the encoded type for this
|
2009-01-20 23:04:12 +03:00
|
|
|
/// property declaration. If non-NULL, Container must be either an
|
2008-08-28 08:38:10 +04:00
|
|
|
/// ObjCCategoryImplDecl or ObjCImplementationDecl; it should only be
|
|
|
|
/// NULL when getting encodings for protocol properties.
|
2009-09-09 19:08:12 +04:00
|
|
|
/// Property attributes are stored as a comma-delimited C string. The simple
|
|
|
|
/// attributes readonly and bycopy are encoded as single characters. The
|
|
|
|
/// parametrized attributes, getter=name, setter=name, and ivar=name, are
|
|
|
|
/// encoded as single characters, followed by an identifier. Property types
|
|
|
|
/// are also encoded as a parametrized attribute. The characters used to encode
|
2009-01-20 23:04:12 +03:00
|
|
|
/// these attributes are defined by the following enumeration:
|
|
|
|
/// @code
|
|
|
|
/// enum PropertyAttributes {
|
|
|
|
/// kPropertyReadOnly = 'R', // property is read-only.
|
|
|
|
/// kPropertyBycopy = 'C', // property is a copy of the value last assigned
|
|
|
|
/// kPropertyByref = '&', // property is a reference to the value last assigned
|
|
|
|
/// kPropertyDynamic = 'D', // property is dynamic
|
|
|
|
/// kPropertyGetter = 'G', // followed by getter selector name
|
|
|
|
/// kPropertySetter = 'S', // followed by setter selector name
|
|
|
|
/// kPropertyInstanceVariable = 'V' // followed by instance variable name
|
2012-03-22 21:48:02 +04:00
|
|
|
/// kPropertyType = 'T' // followed by old-style type encoding.
|
2009-01-20 23:04:12 +03:00
|
|
|
/// kPropertyWeak = 'W' // 'weak' property
|
|
|
|
/// kPropertyStrong = 'P' // property GC'able
|
|
|
|
/// kPropertyNonAtomic = 'N' // property non-atomic
|
|
|
|
/// };
|
|
|
|
/// @endcode
|
2009-09-09 19:08:12 +04:00
|
|
|
void ASTContext::getObjCEncodingForPropertyDecl(const ObjCPropertyDecl *PD,
|
2008-08-28 08:38:10 +04:00
|
|
|
const Decl *Container,
|
2011-01-12 12:06:06 +03:00
|
|
|
std::string& S) const {
|
2008-08-28 08:38:10 +04:00
|
|
|
// Collect information from the property implementation decl(s).
|
|
|
|
bool Dynamic = false;
|
|
|
|
ObjCPropertyImplDecl *SynthesizePID = 0;
|
|
|
|
|
|
|
|
// FIXME: Duplicated code due to poor abstraction.
|
|
|
|
if (Container) {
|
2009-09-09 19:08:12 +04:00
|
|
|
if (const ObjCCategoryImplDecl *CID =
|
2008-08-28 08:38:10 +04:00
|
|
|
dyn_cast<ObjCCategoryImplDecl>(Container)) {
|
|
|
|
for (ObjCCategoryImplDecl::propimpl_iterator
|
2009-06-30 06:36:12 +04:00
|
|
|
i = CID->propimpl_begin(), e = CID->propimpl_end();
|
2009-04-23 05:02:12 +04:00
|
|
|
i != e; ++i) {
|
2008-08-28 08:38:10 +04:00
|
|
|
ObjCPropertyImplDecl *PID = *i;
|
|
|
|
if (PID->getPropertyDecl() == PD) {
|
|
|
|
if (PID->getPropertyImplementation()==ObjCPropertyImplDecl::Dynamic) {
|
|
|
|
Dynamic = true;
|
|
|
|
} else {
|
|
|
|
SynthesizePID = PID;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
} else {
|
2008-10-05 21:34:18 +04:00
|
|
|
const ObjCImplementationDecl *OID=cast<ObjCImplementationDecl>(Container);
|
2008-08-28 08:38:10 +04:00
|
|
|
for (ObjCCategoryImplDecl::propimpl_iterator
|
2009-06-30 06:36:12 +04:00
|
|
|
i = OID->propimpl_begin(), e = OID->propimpl_end();
|
2009-04-23 05:02:12 +04:00
|
|
|
i != e; ++i) {
|
2008-08-28 08:38:10 +04:00
|
|
|
ObjCPropertyImplDecl *PID = *i;
|
|
|
|
if (PID->getPropertyDecl() == PD) {
|
|
|
|
if (PID->getPropertyImplementation()==ObjCPropertyImplDecl::Dynamic) {
|
|
|
|
Dynamic = true;
|
|
|
|
} else {
|
|
|
|
SynthesizePID = PID;
|
|
|
|
}
|
|
|
|
}
|
2009-09-09 19:08:12 +04:00
|
|
|
}
|
2008-08-28 08:38:10 +04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// FIXME: This is not very efficient.
|
|
|
|
S = "T";
|
|
|
|
|
|
|
|
// Encode result type.
|
2009-01-20 22:14:18 +03:00
|
|
|
// GCC has some special rules regarding encoding of properties which
|
|
|
|
// closely resembles encoding of ivars.
|
2009-09-09 19:08:12 +04:00
|
|
|
getObjCEncodingForTypeImpl(PD->getType(), S, true, true, 0,
|
2009-01-20 22:14:18 +03:00
|
|
|
true /* outermost type */,
|
|
|
|
true /* encoding for property */);
|
2008-08-28 08:38:10 +04:00
|
|
|
|
|
|
|
if (PD->isReadOnly()) {
|
|
|
|
S += ",R";
|
|
|
|
} else {
|
|
|
|
switch (PD->getSetterKind()) {
|
|
|
|
case ObjCPropertyDecl::Assign: break;
|
|
|
|
case ObjCPropertyDecl::Copy: S += ",C"; break;
|
2009-09-09 19:08:12 +04:00
|
|
|
case ObjCPropertyDecl::Retain: S += ",&"; break;
|
2011-08-13 00:47:08 +04:00
|
|
|
case ObjCPropertyDecl::Weak: S += ",W"; break;
|
2008-08-28 08:38:10 +04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// It really isn't clear at all what this means, since properties
|
|
|
|
// are "dynamic by default".
|
|
|
|
if (Dynamic)
|
|
|
|
S += ",D";
|
|
|
|
|
2009-01-20 22:14:18 +03:00
|
|
|
if (PD->getPropertyAttributes() & ObjCPropertyDecl::OBJC_PR_nonatomic)
|
|
|
|
S += ",N";
|
2009-09-09 19:08:12 +04:00
|
|
|
|
2008-08-28 08:38:10 +04:00
|
|
|
if (PD->getPropertyAttributes() & ObjCPropertyDecl::OBJC_PR_getter) {
|
|
|
|
S += ",G";
|
2008-11-24 06:33:13 +03:00
|
|
|
S += PD->getGetterName().getAsString();
|
2008-08-28 08:38:10 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
if (PD->getPropertyAttributes() & ObjCPropertyDecl::OBJC_PR_setter) {
|
|
|
|
S += ",S";
|
2008-11-24 06:33:13 +03:00
|
|
|
S += PD->getSetterName().getAsString();
|
2008-08-28 08:38:10 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
if (SynthesizePID) {
|
|
|
|
const ObjCIvarDecl *OID = SynthesizePID->getPropertyIvarDecl();
|
|
|
|
S += ",V";
|
2008-11-24 07:00:27 +03:00
|
|
|
S += OID->getNameAsString();
|
2008-08-28 08:38:10 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
// FIXME: OBJCGC: weak & strong
|
|
|
|
}
|
|
|
|
|
2008-12-23 22:56:47 +03:00
|
|
|
/// getLegacyIntegralTypeEncoding -
|
2009-09-09 19:08:12 +04:00
|
|
|
/// Another legacy compatibility encoding: 32-bit longs are encoded as
|
|
|
|
/// 'l' or 'L' , but not always. For typedefs, we need to use
|
2008-12-23 22:56:47 +03:00
|
|
|
/// 'i' or 'I' instead if encoding a struct field, or a pointer!
|
|
|
|
///
|
|
|
|
void ASTContext::getLegacyIntegralTypeEncoding (QualType &PointeeTy) const {
|
2009-07-22 22:58:19 +04:00
|
|
|
if (isa<TypedefType>(PointeeTy.getTypePtr())) {
|
2009-09-22 03:43:11 +04:00
|
|
|
if (const BuiltinType *BT = PointeeTy->getAs<BuiltinType>()) {
|
2011-01-12 12:06:06 +03:00
|
|
|
if (BT->getKind() == BuiltinType::ULong && getIntWidth(PointeeTy) == 32)
|
2008-12-23 22:56:47 +03:00
|
|
|
PointeeTy = UnsignedIntTy;
|
2009-09-09 19:08:12 +04:00
|
|
|
else
|
2011-01-12 12:06:06 +03:00
|
|
|
if (BT->getKind() == BuiltinType::Long && getIntWidth(PointeeTy) == 32)
|
2008-12-23 22:56:47 +03:00
|
|
|
PointeeTy = IntTy;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2008-01-23 01:44:46 +03:00
|
|
|
void ASTContext::getObjCEncodingForType(QualType T, std::string& S,
|
2011-01-12 12:06:06 +03:00
|
|
|
const FieldDecl *Field) const {
|
2008-10-17 11:30:50 +04:00
|
|
|
// We follow the behavior of gcc, expanding structures which are
|
|
|
|
// directly pointed to, and expanding embedded structures. Note that
|
|
|
|
// these rules are sufficient to prevent recursive encoding of the
|
|
|
|
// same type.
|
2009-09-09 19:08:12 +04:00
|
|
|
getObjCEncodingForTypeImpl(T, S, true, true, Field,
|
2008-12-23 02:22:27 +03:00
|
|
|
true /* outermost type */);
|
2008-10-17 11:30:50 +04:00
|
|
|
}
|
|
|
|
|
2010-06-04 05:10:52 +04:00
|
|
|
static char ObjCEncodingForPrimitiveKind(const ASTContext *C, QualType T) {
|
|
|
|
switch (T->getAs<BuiltinType>()->getKind()) {
|
2011-09-23 09:06:16 +04:00
|
|
|
default: llvm_unreachable("Unhandled builtin type kind");
|
2010-06-04 05:10:52 +04:00
|
|
|
case BuiltinType::Void: return 'v';
|
|
|
|
case BuiltinType::Bool: return 'B';
|
|
|
|
case BuiltinType::Char_U:
|
|
|
|
case BuiltinType::UChar: return 'C';
|
|
|
|
case BuiltinType::UShort: return 'S';
|
|
|
|
case BuiltinType::UInt: return 'I';
|
|
|
|
case BuiltinType::ULong:
|
2011-01-12 12:06:06 +03:00
|
|
|
return C->getIntWidth(T) == 32 ? 'L' : 'Q';
|
2010-06-04 05:10:52 +04:00
|
|
|
case BuiltinType::UInt128: return 'T';
|
|
|
|
case BuiltinType::ULongLong: return 'Q';
|
|
|
|
case BuiltinType::Char_S:
|
|
|
|
case BuiltinType::SChar: return 'c';
|
|
|
|
case BuiltinType::Short: return 's';
|
2010-12-26 02:25:43 +03:00
|
|
|
case BuiltinType::WChar_S:
|
|
|
|
case BuiltinType::WChar_U:
|
2010-06-04 05:10:52 +04:00
|
|
|
case BuiltinType::Int: return 'i';
|
|
|
|
case BuiltinType::Long:
|
2011-01-12 12:06:06 +03:00
|
|
|
return C->getIntWidth(T) == 32 ? 'l' : 'q';
|
2010-06-04 05:10:52 +04:00
|
|
|
case BuiltinType::LongLong: return 'q';
|
|
|
|
case BuiltinType::Int128: return 't';
|
|
|
|
case BuiltinType::Float: return 'f';
|
|
|
|
case BuiltinType::Double: return 'd';
|
2010-10-12 01:13:48 +04:00
|
|
|
case BuiltinType::LongDouble: return 'D';
|
2010-06-04 05:10:52 +04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-09-08 21:18:35 +04:00
|
|
|
static char ObjCEncodingForEnumType(const ASTContext *C, const EnumType *ET) {
|
|
|
|
EnumDecl *Enum = ET->getDecl();
|
|
|
|
|
|
|
|
// The encoding of an non-fixed enum type is always 'i', regardless of size.
|
|
|
|
if (!Enum->isFixed())
|
|
|
|
return 'i';
|
|
|
|
|
|
|
|
// The encoding of a fixed enum type matches its fixed underlying type.
|
|
|
|
return ObjCEncodingForPrimitiveKind(C, Enum->getIntegerType());
|
|
|
|
}
|
|
|
|
|
2011-01-12 12:06:06 +03:00
|
|
|
static void EncodeBitField(const ASTContext *Ctx, std::string& S,
|
2010-06-04 05:10:52 +04:00
|
|
|
QualType T, const FieldDecl *FD) {
|
2011-10-10 22:28:20 +04:00
|
|
|
assert(FD->isBitField() && "not a bitfield - getObjCEncodingForTypeImpl");
|
2009-01-13 04:18:13 +03:00
|
|
|
S += 'b';
|
2010-06-04 05:10:52 +04:00
|
|
|
// The NeXT runtime encodes bit fields as b followed by the number of bits.
|
|
|
|
// The GNU runtime requires more information; bitfields are encoded as b,
|
|
|
|
// then the offset (in bits) of the first element, then the type of the
|
|
|
|
// bitfield, then the size in bits. For example, in this structure:
|
|
|
|
//
|
|
|
|
// struct
|
|
|
|
// {
|
|
|
|
// int integer;
|
|
|
|
// int flags:2;
|
|
|
|
// };
|
|
|
|
// On a 32-bit system, the encoding for flags would be b2 for the NeXT
|
|
|
|
// runtime, but b32i2 for the GNU runtime. The reason for this extra
|
|
|
|
// information is not especially sensible, but we're stuck with it for
|
|
|
|
// compatibility with GCC, although providing it breaks anything that
|
|
|
|
// actually uses runtime introspection and wants to work on both runtimes...
|
2012-03-11 11:00:24 +04:00
|
|
|
if (!Ctx->getLangOpts().NeXTRuntime) {
|
2010-06-04 05:10:52 +04:00
|
|
|
const RecordDecl *RD = FD->getParent();
|
|
|
|
const ASTRecordLayout &RL = Ctx->getASTRecordLayout(RD);
|
2011-07-07 05:54:01 +04:00
|
|
|
S += llvm::utostr(RL.getFieldOffset(FD->getFieldIndex()));
|
2011-09-08 21:18:35 +04:00
|
|
|
if (const EnumType *ET = T->getAs<EnumType>())
|
|
|
|
S += ObjCEncodingForEnumType(Ctx, ET);
|
2010-12-26 23:12:30 +03:00
|
|
|
else
|
2011-01-12 12:06:06 +03:00
|
|
|
S += ObjCEncodingForPrimitiveKind(Ctx, T);
|
2010-06-04 05:10:52 +04:00
|
|
|
}
|
2011-10-10 22:28:20 +04:00
|
|
|
S += llvm::utostr(FD->getBitWidthValue(*Ctx));
|
2009-01-13 04:18:13 +03:00
|
|
|
}
|
|
|
|
|
2009-10-19 01:17:35 +04:00
|
|
|
// FIXME: Use SmallString for accumulating string.
|
2008-10-17 11:30:50 +04:00
|
|
|
void ASTContext::getObjCEncodingForTypeImpl(QualType T, std::string& S,
|
|
|
|
bool ExpandPointedToStructures,
|
|
|
|
bool ExpandStructures,
|
2009-04-20 10:37:24 +04:00
|
|
|
const FieldDecl *FD,
|
2009-01-20 22:14:18 +03:00
|
|
|
bool OutermostType,
|
2011-05-17 04:46:38 +04:00
|
|
|
bool EncodingProperty,
|
2011-11-30 05:57:58 +04:00
|
|
|
bool StructField,
|
|
|
|
bool EncodeBlockParameters,
|
|
|
|
bool EncodeClassNames) const {
|
2010-06-04 05:10:52 +04:00
|
|
|
if (T->getAs<BuiltinType>()) {
|
2009-07-13 04:10:46 +04:00
|
|
|
if (FD && FD->isBitField())
|
2010-06-04 05:10:52 +04:00
|
|
|
return EncodeBitField(this, S, T, FD);
|
|
|
|
S += ObjCEncodingForPrimitiveKind(this, T);
|
2009-07-13 04:10:46 +04:00
|
|
|
return;
|
|
|
|
}
|
2009-09-09 19:08:12 +04:00
|
|
|
|
2009-09-22 03:43:11 +04:00
|
|
|
if (const ComplexType *CT = T->getAs<ComplexType>()) {
|
2009-04-10 01:55:45 +04:00
|
|
|
S += 'j';
|
2009-09-09 19:08:12 +04:00
|
|
|
getObjCEncodingForTypeImpl(CT->getElementType(), S, false, false, 0, false,
|
2009-04-10 01:55:45 +04:00
|
|
|
false);
|
2009-07-13 04:10:46 +04:00
|
|
|
return;
|
|
|
|
}
|
2009-11-23 23:40:50 +03:00
|
|
|
|
2010-04-14 03:45:47 +04:00
|
|
|
// encoding for pointer or r3eference types.
|
|
|
|
QualType PointeeTy;
|
2009-07-30 01:53:49 +04:00
|
|
|
if (const PointerType *PT = T->getAs<PointerType>()) {
|
2009-11-30 21:43:52 +03:00
|
|
|
if (PT->isObjCSelType()) {
|
|
|
|
S += ':';
|
|
|
|
return;
|
|
|
|
}
|
2010-04-14 03:45:47 +04:00
|
|
|
PointeeTy = PT->getPointeeType();
|
|
|
|
}
|
|
|
|
else if (const ReferenceType *RT = T->getAs<ReferenceType>())
|
|
|
|
PointeeTy = RT->getPointeeType();
|
|
|
|
if (!PointeeTy.isNull()) {
|
2008-12-23 22:56:47 +03:00
|
|
|
bool isReadOnly = false;
|
|
|
|
// For historical/compatibility reasons, the read-only qualifier of the
|
|
|
|
// pointee gets emitted _before_ the '^'. The read-only qualifier of
|
|
|
|
// the pointer itself gets ignored, _unless_ we are looking at a typedef!
|
2009-09-09 19:08:12 +04:00
|
|
|
// Also, do not emit the 'r' for anything but the outermost type!
|
2009-07-22 22:58:19 +04:00
|
|
|
if (isa<TypedefType>(T.getTypePtr())) {
|
2008-12-23 22:56:47 +03:00
|
|
|
if (OutermostType && T.isConstQualified()) {
|
|
|
|
isReadOnly = true;
|
|
|
|
S += 'r';
|
|
|
|
}
|
2009-07-31 06:02:20 +04:00
|
|
|
} else if (OutermostType) {
|
2008-12-23 22:56:47 +03:00
|
|
|
QualType P = PointeeTy;
|
2009-07-30 01:53:49 +04:00
|
|
|
while (P->getAs<PointerType>())
|
|
|
|
P = P->getAs<PointerType>()->getPointeeType();
|
2008-12-23 22:56:47 +03:00
|
|
|
if (P.isConstQualified()) {
|
|
|
|
isReadOnly = true;
|
|
|
|
S += 'r';
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (isReadOnly) {
|
|
|
|
// Another legacy compatibility encoding. Some ObjC qualifier and type
|
|
|
|
// combinations need to be rearranged.
|
|
|
|
// Rewrite "in const" from "nr" to "rn"
|
2011-07-23 14:55:15 +04:00
|
|
|
if (StringRef(S).endswith("nr"))
|
2010-04-27 21:12:11 +04:00
|
|
|
S.replace(S.end()-2, S.end(), "rn");
|
2008-12-23 22:56:47 +03:00
|
|
|
}
|
2009-09-09 19:08:12 +04:00
|
|
|
|
2007-10-29 08:01:08 +03:00
|
|
|
if (PointeeTy->isCharType()) {
|
|
|
|
// char pointer types should be encoded as '*' unless it is a
|
|
|
|
// type that has been typedef'd to 'BOOL'.
|
2007-10-29 09:33:42 +03:00
|
|
|
if (!isTypeTypedefedAsBOOL(PointeeTy)) {
|
2007-10-29 08:01:08 +03:00
|
|
|
S += '*';
|
|
|
|
return;
|
|
|
|
}
|
2009-07-30 01:53:49 +04:00
|
|
|
} else if (const RecordType *RTy = PointeeTy->getAs<RecordType>()) {
|
2009-07-22 21:14:51 +04:00
|
|
|
// GCC binary compat: Need to convert "struct objc_class *" to "#".
|
|
|
|
if (RTy->getDecl()->getIdentifier() == &Idents.get("objc_class")) {
|
|
|
|
S += '#';
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
// GCC binary compat: Need to convert "struct objc_object *" to "@".
|
|
|
|
if (RTy->getDecl()->getIdentifier() == &Idents.get("objc_object")) {
|
|
|
|
S += '@';
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
// fall through...
|
2007-10-29 08:01:08 +03:00
|
|
|
}
|
|
|
|
S += '^';
|
2008-12-23 22:56:47 +03:00
|
|
|
getLegacyIntegralTypeEncoding(PointeeTy);
|
|
|
|
|
2009-09-09 19:08:12 +04:00
|
|
|
getObjCEncodingForTypeImpl(PointeeTy, S, false, ExpandPointedToStructures,
|
2008-12-20 02:34:38 +03:00
|
|
|
NULL);
|
2009-07-13 04:10:46 +04:00
|
|
|
return;
|
|
|
|
}
|
2010-04-14 03:45:47 +04:00
|
|
|
|
2009-07-13 04:10:46 +04:00
|
|
|
if (const ArrayType *AT =
|
|
|
|
// Ignore type qualifiers etc.
|
|
|
|
dyn_cast<ArrayType>(T->getCanonicalTypeInternal())) {
|
2011-05-17 04:46:38 +04:00
|
|
|
if (isa<IncompleteArrayType>(AT) && !StructField) {
|
2009-02-22 04:38:57 +03:00
|
|
|
// Incomplete arrays are encoded as a pointer to the array element.
|
|
|
|
S += '^';
|
|
|
|
|
2009-09-09 19:08:12 +04:00
|
|
|
getObjCEncodingForTypeImpl(AT->getElementType(), S,
|
2009-02-22 04:38:57 +03:00
|
|
|
false, ExpandStructures, FD);
|
|
|
|
} else {
|
|
|
|
S += '[';
|
2009-09-09 19:08:12 +04:00
|
|
|
|
2011-05-17 04:46:38 +04:00
|
|
|
if (const ConstantArrayType *CAT = dyn_cast<ConstantArrayType>(AT)) {
|
|
|
|
if (getTypeSize(CAT->getElementType()) == 0)
|
|
|
|
S += '0';
|
|
|
|
else
|
|
|
|
S += llvm::utostr(CAT->getSize().getZExtValue());
|
|
|
|
} else {
|
2009-02-22 04:38:57 +03:00
|
|
|
//Variable length arrays are encoded as a regular array with 0 elements.
|
2011-05-17 04:46:38 +04:00
|
|
|
assert((isa<VariableArrayType>(AT) || isa<IncompleteArrayType>(AT)) &&
|
|
|
|
"Unknown array type!");
|
2009-02-22 04:38:57 +03:00
|
|
|
S += '0';
|
|
|
|
}
|
2009-09-09 19:08:12 +04:00
|
|
|
|
|
|
|
getObjCEncodingForTypeImpl(AT->getElementType(), S,
|
2009-02-22 04:38:57 +03:00
|
|
|
false, ExpandStructures, FD);
|
|
|
|
S += ']';
|
|
|
|
}
|
2009-07-13 04:10:46 +04:00
|
|
|
return;
|
|
|
|
}
|
2009-09-09 19:08:12 +04:00
|
|
|
|
2009-09-22 03:43:11 +04:00
|
|
|
if (T->getAs<FunctionType>()) {
|
2007-10-30 03:06:20 +03:00
|
|
|
S += '?';
|
2009-07-13 04:10:46 +04:00
|
|
|
return;
|
|
|
|
}
|
2009-09-09 19:08:12 +04:00
|
|
|
|
2009-07-30 01:53:49 +04:00
|
|
|
if (const RecordType *RTy = T->getAs<RecordType>()) {
|
2008-10-17 11:30:50 +04:00
|
|
|
RecordDecl *RDecl = RTy->getDecl();
|
2008-10-17 20:17:37 +04:00
|
|
|
S += RDecl->isUnion() ? '(' : '{';
|
2008-10-17 10:22:57 +04:00
|
|
|
// Anonymous structures print as '?'
|
|
|
|
if (const IdentifierInfo *II = RDecl->getIdentifier()) {
|
|
|
|
S += II->getName();
|
2010-05-07 04:28:49 +04:00
|
|
|
if (ClassTemplateSpecializationDecl *Spec
|
|
|
|
= dyn_cast<ClassTemplateSpecializationDecl>(RDecl)) {
|
|
|
|
const TemplateArgumentList &TemplateArgs = Spec->getTemplateArgs();
|
|
|
|
std::string TemplateArgsStr
|
|
|
|
= TemplateSpecializationType::PrintTemplateArgumentList(
|
2010-11-08 02:05:16 +03:00
|
|
|
TemplateArgs.data(),
|
|
|
|
TemplateArgs.size(),
|
2011-09-28 02:38:19 +04:00
|
|
|
(*this).getPrintingPolicy());
|
2010-05-07 04:28:49 +04:00
|
|
|
|
|
|
|
S += TemplateArgsStr;
|
|
|
|
}
|
2008-10-17 10:22:57 +04:00
|
|
|
} else {
|
|
|
|
S += '?';
|
|
|
|
}
|
2008-10-18 00:21:44 +04:00
|
|
|
if (ExpandStructures) {
|
2008-01-23 01:44:46 +03:00
|
|
|
S += '=';
|
2011-05-17 04:46:38 +04:00
|
|
|
if (!RDecl->isUnion()) {
|
|
|
|
getObjCEncodingForStructureImpl(RDecl, S, FD);
|
|
|
|
} else {
|
|
|
|
for (RecordDecl::field_iterator Field = RDecl->field_begin(),
|
|
|
|
FieldEnd = RDecl->field_end();
|
|
|
|
Field != FieldEnd; ++Field) {
|
|
|
|
if (FD) {
|
|
|
|
S += '"';
|
|
|
|
S += Field->getNameAsString();
|
|
|
|
S += '"';
|
|
|
|
}
|
2009-09-09 19:08:12 +04:00
|
|
|
|
2011-05-17 04:46:38 +04:00
|
|
|
// Special case bit-fields.
|
|
|
|
if (Field->isBitField()) {
|
|
|
|
getObjCEncodingForTypeImpl(Field->getType(), S, false, true,
|
|
|
|
(*Field));
|
|
|
|
} else {
|
|
|
|
QualType qt = Field->getType();
|
|
|
|
getLegacyIntegralTypeEncoding(qt);
|
|
|
|
getObjCEncodingForTypeImpl(qt, S, false, true,
|
|
|
|
FD, /*OutermostType*/false,
|
|
|
|
/*EncodingProperty*/false,
|
|
|
|
/*StructField*/true);
|
|
|
|
}
|
2008-10-17 20:17:37 +04:00
|
|
|
}
|
2008-01-23 01:44:46 +03:00
|
|
|
}
|
2007-11-14 02:21:38 +03:00
|
|
|
}
|
2008-10-17 20:17:37 +04:00
|
|
|
S += RDecl->isUnion() ? ')' : '}';
|
2009-07-13 04:10:46 +04:00
|
|
|
return;
|
|
|
|
}
|
2010-10-08 01:25:25 +04:00
|
|
|
|
2011-09-08 21:18:35 +04:00
|
|
|
if (const EnumType *ET = T->getAs<EnumType>()) {
|
2009-01-13 04:18:13 +03:00
|
|
|
if (FD && FD->isBitField())
|
2010-06-04 05:10:52 +04:00
|
|
|
EncodeBitField(this, S, T, FD);
|
2009-01-13 04:18:13 +03:00
|
|
|
else
|
2011-09-08 21:18:35 +04:00
|
|
|
S += ObjCEncodingForEnumType(this, ET);
|
2009-07-13 04:10:46 +04:00
|
|
|
return;
|
|
|
|
}
|
2009-09-09 19:08:12 +04:00
|
|
|
|
2011-11-30 05:57:58 +04:00
|
|
|
if (const BlockPointerType *BT = T->getAs<BlockPointerType>()) {
|
2009-02-02 21:24:29 +03:00
|
|
|
S += "@?"; // Unlike a pointer-to-function, which is "^?".
|
2011-11-30 05:57:58 +04:00
|
|
|
if (EncodeBlockParameters) {
|
|
|
|
const FunctionType *FT = BT->getPointeeType()->getAs<FunctionType>();
|
|
|
|
|
|
|
|
S += '<';
|
|
|
|
// Block return type
|
|
|
|
getObjCEncodingForTypeImpl(FT->getResultType(), S,
|
|
|
|
ExpandPointedToStructures, ExpandStructures,
|
|
|
|
FD,
|
|
|
|
false /* OutermostType */,
|
|
|
|
EncodingProperty,
|
|
|
|
false /* StructField */,
|
|
|
|
EncodeBlockParameters,
|
|
|
|
EncodeClassNames);
|
|
|
|
// Block self
|
|
|
|
S += "@?";
|
|
|
|
// Block parameters
|
|
|
|
if (const FunctionProtoType *FPT = dyn_cast<FunctionProtoType>(FT)) {
|
|
|
|
for (FunctionProtoType::arg_type_iterator I = FPT->arg_type_begin(),
|
|
|
|
E = FPT->arg_type_end(); I && (I != E); ++I) {
|
|
|
|
getObjCEncodingForTypeImpl(*I, S,
|
|
|
|
ExpandPointedToStructures,
|
|
|
|
ExpandStructures,
|
|
|
|
FD,
|
|
|
|
false /* OutermostType */,
|
|
|
|
EncodingProperty,
|
|
|
|
false /* StructField */,
|
|
|
|
EncodeBlockParameters,
|
|
|
|
EncodeClassNames);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
S += '>';
|
|
|
|
}
|
2009-07-13 04:10:46 +04:00
|
|
|
return;
|
|
|
|
}
|
2009-09-09 19:08:12 +04:00
|
|
|
|
2010-05-15 15:32:37 +04:00
|
|
|
// Ignore protocol qualifiers when mangling at this level.
|
|
|
|
if (const ObjCObjectType *OT = T->getAs<ObjCObjectType>())
|
|
|
|
T = OT->getBaseType();
|
|
|
|
|
2009-09-24 23:53:00 +04:00
|
|
|
if (const ObjCInterfaceType *OIT = T->getAs<ObjCInterfaceType>()) {
|
2008-12-20 02:34:38 +03:00
|
|
|
// @encode(class_name)
|
2009-09-24 23:53:00 +04:00
|
|
|
ObjCInterfaceDecl *OI = OIT->getDecl();
|
2008-12-20 02:34:38 +03:00
|
|
|
S += '{';
|
|
|
|
const IdentifierInfo *II = OI->getIdentifier();
|
|
|
|
S += II->getName();
|
|
|
|
S += '=';
|
2011-07-22 06:08:32 +04:00
|
|
|
SmallVector<const ObjCIvarDecl*, 32> Ivars;
|
2010-08-21 01:21:08 +04:00
|
|
|
DeepCollectObjCIvars(OI, true, Ivars);
|
|
|
|
for (unsigned i = 0, e = Ivars.size(); i != e; ++i) {
|
2011-07-22 06:08:32 +04:00
|
|
|
const FieldDecl *Field = cast<FieldDecl>(Ivars[i]);
|
2010-08-21 01:21:08 +04:00
|
|
|
if (Field->isBitField())
|
|
|
|
getObjCEncodingForTypeImpl(Field->getType(), S, false, true, Field);
|
2008-12-20 02:34:38 +03:00
|
|
|
else
|
2010-08-21 01:21:08 +04:00
|
|
|
getObjCEncodingForTypeImpl(Field->getType(), S, false, true, FD);
|
2008-12-20 02:34:38 +03:00
|
|
|
}
|
|
|
|
S += '}';
|
2009-07-13 04:10:46 +04:00
|
|
|
return;
|
2008-12-20 02:34:38 +03:00
|
|
|
}
|
2009-09-09 19:08:12 +04:00
|
|
|
|
2009-09-22 03:43:11 +04:00
|
|
|
if (const ObjCObjectPointerType *OPT = T->getAs<ObjCObjectPointerType>()) {
|
2009-07-11 03:34:53 +04:00
|
|
|
if (OPT->isObjCIdType()) {
|
|
|
|
S += '@';
|
|
|
|
return;
|
2009-07-13 04:10:46 +04:00
|
|
|
}
|
2009-09-09 19:08:12 +04:00
|
|
|
|
2009-10-29 01:03:49 +03:00
|
|
|
if (OPT->isObjCClassType() || OPT->isObjCQualifiedClassType()) {
|
|
|
|
// FIXME: Consider if we need to output qualifiers for 'Class<p>'.
|
|
|
|
// Since this is a binary compatibility issue, need to consult with runtime
|
|
|
|
// folks. Fortunately, this is a *very* obsure construct.
|
2009-07-11 03:34:53 +04:00
|
|
|
S += '#';
|
|
|
|
return;
|
2009-07-13 04:10:46 +04:00
|
|
|
}
|
2009-09-09 19:08:12 +04:00
|
|
|
|
2009-07-13 04:10:46 +04:00
|
|
|
if (OPT->isObjCQualifiedIdType()) {
|
2009-09-09 19:08:12 +04:00
|
|
|
getObjCEncodingForTypeImpl(getObjCIdType(), S,
|
2009-07-11 03:34:53 +04:00
|
|
|
ExpandPointedToStructures,
|
|
|
|
ExpandStructures, FD);
|
2011-11-30 05:57:58 +04:00
|
|
|
if (FD || EncodingProperty || EncodeClassNames) {
|
2009-07-11 03:34:53 +04:00
|
|
|
// Note that we do extended encoding of protocol qualifer list
|
|
|
|
// Only when doing ivar or property encoding.
|
|
|
|
S += '"';
|
2009-07-20 21:56:53 +04:00
|
|
|
for (ObjCObjectPointerType::qual_iterator I = OPT->qual_begin(),
|
|
|
|
E = OPT->qual_end(); I != E; ++I) {
|
2009-07-11 03:34:53 +04:00
|
|
|
S += '<';
|
|
|
|
S += (*I)->getNameAsString();
|
|
|
|
S += '>';
|
|
|
|
}
|
|
|
|
S += '"';
|
|
|
|
}
|
|
|
|
return;
|
2009-07-13 04:10:46 +04:00
|
|
|
}
|
2009-09-09 19:08:12 +04:00
|
|
|
|
2009-07-13 04:10:46 +04:00
|
|
|
QualType PointeeTy = OPT->getPointeeType();
|
|
|
|
if (!EncodingProperty &&
|
|
|
|
isa<TypedefType>(PointeeTy.getTypePtr())) {
|
|
|
|
// Another historical/compatibility reason.
|
2009-09-09 19:08:12 +04:00
|
|
|
// We encode the underlying type which comes out as
|
2009-07-13 04:10:46 +04:00
|
|
|
// {...};
|
|
|
|
S += '^';
|
2009-09-09 19:08:12 +04:00
|
|
|
getObjCEncodingForTypeImpl(PointeeTy, S,
|
|
|
|
false, ExpandPointedToStructures,
|
2009-07-13 04:10:46 +04:00
|
|
|
NULL);
|
2009-07-11 03:34:53 +04:00
|
|
|
return;
|
|
|
|
}
|
2009-07-13 04:10:46 +04:00
|
|
|
|
|
|
|
S += '@';
|
2011-11-30 05:57:58 +04:00
|
|
|
if (OPT->getInterfaceDecl() &&
|
|
|
|
(FD || EncodingProperty || EncodeClassNames)) {
|
2009-07-13 04:10:46 +04:00
|
|
|
S += '"';
|
2009-10-29 01:03:49 +03:00
|
|
|
S += OPT->getInterfaceDecl()->getIdentifier()->getName();
|
2009-07-20 21:56:53 +04:00
|
|
|
for (ObjCObjectPointerType::qual_iterator I = OPT->qual_begin(),
|
|
|
|
E = OPT->qual_end(); I != E; ++I) {
|
2009-07-13 04:10:46 +04:00
|
|
|
S += '<';
|
|
|
|
S += (*I)->getNameAsString();
|
|
|
|
S += '>';
|
2009-09-09 19:08:12 +04:00
|
|
|
}
|
2009-07-13 04:10:46 +04:00
|
|
|
S += '"';
|
|
|
|
}
|
|
|
|
return;
|
|
|
|
}
|
2009-09-09 19:08:12 +04:00
|
|
|
|
2010-05-18 03:56:34 +04:00
|
|
|
// gcc just blithely ignores member pointers.
|
|
|
|
// TODO: maybe there should be a mangling for these
|
|
|
|
if (T->getAs<MemberPointerType>())
|
|
|
|
return;
|
2010-10-08 01:25:25 +04:00
|
|
|
|
|
|
|
if (T->isVectorType()) {
|
|
|
|
// This matches gcc's encoding, even though technically it is
|
|
|
|
// insufficient.
|
|
|
|
// FIXME. We should do a better job than gcc.
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2011-09-23 09:06:16 +04:00
|
|
|
llvm_unreachable("@encode for type not implemented!");
|
2007-10-29 08:01:08 +03:00
|
|
|
}
|
|
|
|
|
2011-05-17 04:46:38 +04:00
|
|
|
void ASTContext::getObjCEncodingForStructureImpl(RecordDecl *RDecl,
|
|
|
|
std::string &S,
|
|
|
|
const FieldDecl *FD,
|
|
|
|
bool includeVBases) const {
|
|
|
|
assert(RDecl && "Expected non-null RecordDecl");
|
|
|
|
assert(!RDecl->isUnion() && "Should not be called for unions");
|
|
|
|
if (!RDecl->getDefinition())
|
|
|
|
return;
|
|
|
|
|
|
|
|
CXXRecordDecl *CXXRec = dyn_cast<CXXRecordDecl>(RDecl);
|
|
|
|
std::multimap<uint64_t, NamedDecl *> FieldOrBaseOffsets;
|
|
|
|
const ASTRecordLayout &layout = getASTRecordLayout(RDecl);
|
|
|
|
|
|
|
|
if (CXXRec) {
|
|
|
|
for (CXXRecordDecl::base_class_iterator
|
|
|
|
BI = CXXRec->bases_begin(),
|
|
|
|
BE = CXXRec->bases_end(); BI != BE; ++BI) {
|
|
|
|
if (!BI->isVirtual()) {
|
|
|
|
CXXRecordDecl *base = BI->getType()->getAsCXXRecordDecl();
|
2011-06-18 03:19:38 +04:00
|
|
|
if (base->isEmpty())
|
|
|
|
continue;
|
2011-05-17 04:46:38 +04:00
|
|
|
uint64_t offs = layout.getBaseClassOffsetInBits(base);
|
|
|
|
FieldOrBaseOffsets.insert(FieldOrBaseOffsets.upper_bound(offs),
|
|
|
|
std::make_pair(offs, base));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
unsigned i = 0;
|
|
|
|
for (RecordDecl::field_iterator Field = RDecl->field_begin(),
|
|
|
|
FieldEnd = RDecl->field_end();
|
|
|
|
Field != FieldEnd; ++Field, ++i) {
|
|
|
|
uint64_t offs = layout.getFieldOffset(i);
|
|
|
|
FieldOrBaseOffsets.insert(FieldOrBaseOffsets.upper_bound(offs),
|
|
|
|
std::make_pair(offs, *Field));
|
|
|
|
}
|
|
|
|
|
|
|
|
if (CXXRec && includeVBases) {
|
|
|
|
for (CXXRecordDecl::base_class_iterator
|
|
|
|
BI = CXXRec->vbases_begin(),
|
|
|
|
BE = CXXRec->vbases_end(); BI != BE; ++BI) {
|
|
|
|
CXXRecordDecl *base = BI->getType()->getAsCXXRecordDecl();
|
2011-06-18 03:19:38 +04:00
|
|
|
if (base->isEmpty())
|
|
|
|
continue;
|
2011-05-17 04:46:38 +04:00
|
|
|
uint64_t offs = layout.getVBaseClassOffsetInBits(base);
|
2011-09-26 22:14:24 +04:00
|
|
|
if (FieldOrBaseOffsets.find(offs) == FieldOrBaseOffsets.end())
|
|
|
|
FieldOrBaseOffsets.insert(FieldOrBaseOffsets.end(),
|
|
|
|
std::make_pair(offs, base));
|
2011-05-17 04:46:38 +04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
CharUnits size;
|
|
|
|
if (CXXRec) {
|
|
|
|
size = includeVBases ? layout.getSize() : layout.getNonVirtualSize();
|
|
|
|
} else {
|
|
|
|
size = layout.getSize();
|
|
|
|
}
|
|
|
|
|
|
|
|
uint64_t CurOffs = 0;
|
|
|
|
std::multimap<uint64_t, NamedDecl *>::iterator
|
|
|
|
CurLayObj = FieldOrBaseOffsets.begin();
|
|
|
|
|
2011-08-22 20:03:14 +04:00
|
|
|
if ((CurLayObj != FieldOrBaseOffsets.end() && CurLayObj->first != 0) ||
|
|
|
|
(CurLayObj == FieldOrBaseOffsets.end() &&
|
|
|
|
CXXRec && CXXRec->isDynamicClass())) {
|
2011-05-17 04:46:38 +04:00
|
|
|
assert(CXXRec && CXXRec->isDynamicClass() &&
|
|
|
|
"Offset 0 was empty but no VTable ?");
|
|
|
|
if (FD) {
|
|
|
|
S += "\"_vptr$";
|
|
|
|
std::string recname = CXXRec->getNameAsString();
|
|
|
|
if (recname.empty()) recname = "?";
|
|
|
|
S += recname;
|
|
|
|
S += '"';
|
|
|
|
}
|
|
|
|
S += "^^?";
|
|
|
|
CurOffs += getTypeSize(VoidPtrTy);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!RDecl->hasFlexibleArrayMember()) {
|
|
|
|
// Mark the end of the structure.
|
|
|
|
uint64_t offs = toBits(size);
|
|
|
|
FieldOrBaseOffsets.insert(FieldOrBaseOffsets.upper_bound(offs),
|
|
|
|
std::make_pair(offs, (NamedDecl*)0));
|
|
|
|
}
|
|
|
|
|
|
|
|
for (; CurLayObj != FieldOrBaseOffsets.end(); ++CurLayObj) {
|
|
|
|
assert(CurOffs <= CurLayObj->first);
|
|
|
|
|
|
|
|
if (CurOffs < CurLayObj->first) {
|
|
|
|
uint64_t padding = CurLayObj->first - CurOffs;
|
|
|
|
// FIXME: There doesn't seem to be a way to indicate in the encoding that
|
|
|
|
// packing/alignment of members is different that normal, in which case
|
|
|
|
// the encoding will be out-of-sync with the real layout.
|
|
|
|
// If the runtime switches to just consider the size of types without
|
|
|
|
// taking into account alignment, we could make padding explicit in the
|
|
|
|
// encoding (e.g. using arrays of chars). The encoding strings would be
|
|
|
|
// longer then though.
|
|
|
|
CurOffs += padding;
|
|
|
|
}
|
|
|
|
|
|
|
|
NamedDecl *dcl = CurLayObj->second;
|
|
|
|
if (dcl == 0)
|
|
|
|
break; // reached end of structure.
|
|
|
|
|
|
|
|
if (CXXRecordDecl *base = dyn_cast<CXXRecordDecl>(dcl)) {
|
|
|
|
// We expand the bases without their virtual bases since those are going
|
|
|
|
// in the initial structure. Note that this differs from gcc which
|
|
|
|
// expands virtual bases each time one is encountered in the hierarchy,
|
|
|
|
// making the encoding type bigger than it really is.
|
|
|
|
getObjCEncodingForStructureImpl(base, S, FD, /*includeVBases*/false);
|
2011-06-18 03:19:38 +04:00
|
|
|
assert(!base->isEmpty());
|
|
|
|
CurOffs += toBits(getASTRecordLayout(base).getNonVirtualSize());
|
2011-05-17 04:46:38 +04:00
|
|
|
} else {
|
|
|
|
FieldDecl *field = cast<FieldDecl>(dcl);
|
|
|
|
if (FD) {
|
|
|
|
S += '"';
|
|
|
|
S += field->getNameAsString();
|
|
|
|
S += '"';
|
|
|
|
}
|
|
|
|
|
|
|
|
if (field->isBitField()) {
|
|
|
|
EncodeBitField(this, S, field->getType(), field);
|
2011-10-10 22:28:20 +04:00
|
|
|
CurOffs += field->getBitWidthValue(*this);
|
2011-05-17 04:46:38 +04:00
|
|
|
} else {
|
|
|
|
QualType qt = field->getType();
|
|
|
|
getLegacyIntegralTypeEncoding(qt);
|
|
|
|
getObjCEncodingForTypeImpl(qt, S, false, true, FD,
|
|
|
|
/*OutermostType*/false,
|
|
|
|
/*EncodingProperty*/false,
|
|
|
|
/*StructField*/true);
|
|
|
|
CurOffs += getTypeSize(field->getType());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2009-09-09 19:08:12 +04:00
|
|
|
void ASTContext::getObjCEncodingForTypeQualifier(Decl::ObjCDeclQualifier QT,
|
2007-11-01 20:18:37 +03:00
|
|
|
std::string& S) const {
|
|
|
|
if (QT & Decl::OBJC_TQ_In)
|
|
|
|
S += 'n';
|
|
|
|
if (QT & Decl::OBJC_TQ_Inout)
|
|
|
|
S += 'N';
|
|
|
|
if (QT & Decl::OBJC_TQ_Out)
|
|
|
|
S += 'o';
|
|
|
|
if (QT & Decl::OBJC_TQ_Bycopy)
|
|
|
|
S += 'O';
|
|
|
|
if (QT & Decl::OBJC_TQ_Byref)
|
|
|
|
S += 'R';
|
|
|
|
if (QT & Decl::OBJC_TQ_Oneway)
|
|
|
|
S += 'V';
|
|
|
|
}
|
|
|
|
|
2009-07-13 04:10:46 +04:00
|
|
|
void ASTContext::setBuiltinVaListType(QualType T) {
|
2007-10-11 05:00:40 +04:00
|
|
|
assert(BuiltinVaListType.isNull() && "__builtin_va_list type already set!");
|
2009-09-09 19:08:12 +04:00
|
|
|
|
2007-10-11 05:00:40 +04:00
|
|
|
BuiltinVaListType = T;
|
|
|
|
}
|
|
|
|
|
2011-08-12 09:46:01 +04:00
|
|
|
TypedefDecl *ASTContext::getObjCIdDecl() const {
|
|
|
|
if (!ObjCIdDecl) {
|
|
|
|
QualType T = getObjCObjectType(ObjCBuiltinIdTy, 0, 0);
|
|
|
|
T = getObjCObjectPointerType(T);
|
|
|
|
TypeSourceInfo *IdInfo = getTrivialTypeSourceInfo(T);
|
|
|
|
ObjCIdDecl = TypedefDecl::Create(const_cast<ASTContext &>(*this),
|
|
|
|
getTranslationUnitDecl(),
|
|
|
|
SourceLocation(), SourceLocation(),
|
|
|
|
&Idents.get("id"), IdInfo);
|
|
|
|
}
|
|
|
|
|
|
|
|
return ObjCIdDecl;
|
2007-10-15 18:41:52 +04:00
|
|
|
}
|
|
|
|
|
2011-08-12 10:17:30 +04:00
|
|
|
TypedefDecl *ASTContext::getObjCSelDecl() const {
|
|
|
|
if (!ObjCSelDecl) {
|
|
|
|
QualType SelT = getPointerType(ObjCBuiltinSelTy);
|
|
|
|
TypeSourceInfo *SelInfo = getTrivialTypeSourceInfo(SelT);
|
|
|
|
ObjCSelDecl = TypedefDecl::Create(const_cast<ASTContext &>(*this),
|
|
|
|
getTranslationUnitDecl(),
|
|
|
|
SourceLocation(), SourceLocation(),
|
|
|
|
&Idents.get("SEL"), SelInfo);
|
|
|
|
}
|
|
|
|
return ObjCSelDecl;
|
2007-10-17 00:40:23 +04:00
|
|
|
}
|
|
|
|
|
2011-08-12 09:59:41 +04:00
|
|
|
TypedefDecl *ASTContext::getObjCClassDecl() const {
|
|
|
|
if (!ObjCClassDecl) {
|
|
|
|
QualType T = getObjCObjectType(ObjCBuiltinClassTy, 0, 0);
|
|
|
|
T = getObjCObjectPointerType(T);
|
|
|
|
TypeSourceInfo *ClassInfo = getTrivialTypeSourceInfo(T);
|
|
|
|
ObjCClassDecl = TypedefDecl::Create(const_cast<ASTContext &>(*this),
|
|
|
|
getTranslationUnitDecl(),
|
|
|
|
SourceLocation(), SourceLocation(),
|
|
|
|
&Idents.get("Class"), ClassInfo);
|
|
|
|
}
|
|
|
|
|
|
|
|
return ObjCClassDecl;
|
2012-01-17 22:09:05 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
ObjCInterfaceDecl *ASTContext::getObjCProtocolDecl() const {
|
|
|
|
if (!ObjCProtocolClassDecl) {
|
|
|
|
ObjCProtocolClassDecl
|
|
|
|
= ObjCInterfaceDecl::Create(*this, getTranslationUnitDecl(),
|
|
|
|
SourceLocation(),
|
|
|
|
&Idents.get("Protocol"),
|
|
|
|
/*PrevDecl=*/0,
|
|
|
|
SourceLocation(), true);
|
|
|
|
}
|
|
|
|
|
|
|
|
return ObjCProtocolClassDecl;
|
2007-10-31 05:53:19 +03:00
|
|
|
}
|
|
|
|
|
2008-01-07 22:49:32 +03:00
|
|
|
void ASTContext::setObjCConstantStringInterface(ObjCInterfaceDecl *Decl) {
|
2009-09-09 19:08:12 +04:00
|
|
|
assert(ObjCConstantStringType.isNull() &&
|
2007-10-16 03:35:17 +04:00
|
|
|
"'NSConstantString' type already set!");
|
2009-09-09 19:08:12 +04:00
|
|
|
|
2008-01-07 22:49:32 +03:00
|
|
|
ObjCConstantStringType = getObjCInterfaceType(Decl);
|
2007-10-16 03:35:17 +04:00
|
|
|
}
|
|
|
|
|
2009-12-02 11:04:21 +03:00
|
|
|
/// \brief Retrieve the template name that corresponds to a non-empty
|
|
|
|
/// lookup.
|
2011-01-12 12:06:06 +03:00
|
|
|
TemplateName
|
|
|
|
ASTContext::getOverloadedTemplateName(UnresolvedSetIterator Begin,
|
|
|
|
UnresolvedSetIterator End) const {
|
2009-12-02 11:04:21 +03:00
|
|
|
unsigned size = End - Begin;
|
|
|
|
assert(size > 1 && "set is not overloaded!");
|
2009-03-31 02:58:21 +04:00
|
|
|
|
2009-12-02 11:04:21 +03:00
|
|
|
void *memory = Allocate(sizeof(OverloadedTemplateStorage) +
|
|
|
|
size * sizeof(FunctionTemplateDecl*));
|
|
|
|
OverloadedTemplateStorage *OT = new(memory) OverloadedTemplateStorage(size);
|
|
|
|
|
|
|
|
NamedDecl **Storage = OT->getStorage();
|
2010-01-20 03:46:10 +03:00
|
|
|
for (UnresolvedSetIterator I = Begin; I != End; ++I) {
|
2009-12-02 11:04:21 +03:00
|
|
|
NamedDecl *D = *I;
|
|
|
|
assert(isa<FunctionTemplateDecl>(D) ||
|
|
|
|
(isa<UsingShadowDecl>(D) &&
|
|
|
|
isa<FunctionTemplateDecl>(D->getUnderlyingDecl())));
|
|
|
|
*Storage++ = D;
|
2009-03-31 02:58:21 +04:00
|
|
|
}
|
|
|
|
|
2009-12-02 11:04:21 +03:00
|
|
|
return TemplateName(OT);
|
2009-03-31 02:58:21 +04:00
|
|
|
}
|
|
|
|
|
2009-07-29 22:26:50 +04:00
|
|
|
/// \brief Retrieve the template name that represents a qualified
|
|
|
|
/// template name such as \c std::vector.
|
2011-01-12 12:06:06 +03:00
|
|
|
TemplateName
|
|
|
|
ASTContext::getQualifiedTemplateName(NestedNameSpecifier *NNS,
|
|
|
|
bool TemplateKeyword,
|
|
|
|
TemplateDecl *Template) const {
|
2011-03-03 20:04:51 +03:00
|
|
|
assert(NNS && "Missing nested-name-specifier in qualified template name");
|
|
|
|
|
Fix an obscure crash found in the Boost.MPL test suite, along with a
ton of potential crashes of the same kind. The fundamental problem is
that type creation was following a dangerous pattern when using its
FoldingSets:
1) Use FindNodeOrInsertPos to see if the type is available
2) If not, and we aren't looking at a canonical type, build the
canonical type
3) Build and insert the new node into the FoldingSet
The problem here is that building the canonical type can, in very rare
circumstances, force the hash table inside the FoldingSet to
reallocate. That invalidates the insertion position we computed in
step 1, and in step 3 we end up inserting the new node into the wrong
place. BOOM!
I've audited all of ASTContext, fixing this problem everywhere I found
it. The vast majority of wrong code was C++-specific (and *ahem*
written by me), so I also audited other major folding sets in the C++
code (e.g., template specializations), but found no other instances of
this problem.
git-svn-id: https://llvm.org/svn/llvm-project/cfe/trunk@95315 91177308-0d34-0410-b5e6-96231b3b80d8
2010-02-04 21:10:26 +03:00
|
|
|
// FIXME: Canonicalization?
|
2009-07-29 22:26:50 +04:00
|
|
|
llvm::FoldingSetNodeID ID;
|
|
|
|
QualifiedTemplateName::Profile(ID, NNS, TemplateKeyword, Template);
|
2009-09-09 19:08:12 +04:00
|
|
|
|
2009-07-29 22:26:50 +04:00
|
|
|
void *InsertPos = 0;
|
|
|
|
QualifiedTemplateName *QTN =
|
2009-12-02 11:04:21 +03:00
|
|
|
QualifiedTemplateNames.FindNodeOrInsertPos(ID, InsertPos);
|
2009-07-29 22:26:50 +04:00
|
|
|
if (!QTN) {
|
|
|
|
QTN = new (*this,4) QualifiedTemplateName(NNS, TemplateKeyword, Template);
|
|
|
|
QualifiedTemplateNames.InsertNode(QTN, InsertPos);
|
|
|
|
}
|
2009-09-09 19:08:12 +04:00
|
|
|
|
2009-07-29 22:26:50 +04:00
|
|
|
return TemplateName(QTN);
|
|
|
|
}
|
|
|
|
|
2009-03-31 02:58:21 +04:00
|
|
|
/// \brief Retrieve the template name that represents a dependent
|
|
|
|
/// template name such as \c MetaFun::template apply.
|
2011-01-12 12:06:06 +03:00
|
|
|
TemplateName
|
|
|
|
ASTContext::getDependentTemplateName(NestedNameSpecifier *NNS,
|
|
|
|
const IdentifierInfo *Name) const {
|
2009-09-09 19:08:12 +04:00
|
|
|
assert((!NNS || NNS->isDependent()) &&
|
2009-09-09 04:23:06 +04:00
|
|
|
"Nested name specifier must be dependent");
|
2009-03-31 02:58:21 +04:00
|
|
|
|
|
|
|
llvm::FoldingSetNodeID ID;
|
|
|
|
DependentTemplateName::Profile(ID, NNS, Name);
|
|
|
|
|
|
|
|
void *InsertPos = 0;
|
|
|
|
DependentTemplateName *QTN =
|
|
|
|
DependentTemplateNames.FindNodeOrInsertPos(ID, InsertPos);
|
|
|
|
|
|
|
|
if (QTN)
|
|
|
|
return TemplateName(QTN);
|
|
|
|
|
|
|
|
NestedNameSpecifier *CanonNNS = getCanonicalNestedNameSpecifier(NNS);
|
|
|
|
if (CanonNNS == NNS) {
|
|
|
|
QTN = new (*this,4) DependentTemplateName(NNS, Name);
|
|
|
|
} else {
|
|
|
|
TemplateName Canon = getDependentTemplateName(CanonNNS, Name);
|
|
|
|
QTN = new (*this,4) DependentTemplateName(NNS, Name, Canon);
|
Fix an obscure crash found in the Boost.MPL test suite, along with a
ton of potential crashes of the same kind. The fundamental problem is
that type creation was following a dangerous pattern when using its
FoldingSets:
1) Use FindNodeOrInsertPos to see if the type is available
2) If not, and we aren't looking at a canonical type, build the
canonical type
3) Build and insert the new node into the FoldingSet
The problem here is that building the canonical type can, in very rare
circumstances, force the hash table inside the FoldingSet to
reallocate. That invalidates the insertion position we computed in
step 1, and in step 3 we end up inserting the new node into the wrong
place. BOOM!
I've audited all of ASTContext, fixing this problem everywhere I found
it. The vast majority of wrong code was C++-specific (and *ahem*
written by me), so I also audited other major folding sets in the C++
code (e.g., template specializations), but found no other instances of
this problem.
git-svn-id: https://llvm.org/svn/llvm-project/cfe/trunk@95315 91177308-0d34-0410-b5e6-96231b3b80d8
2010-02-04 21:10:26 +03:00
|
|
|
DependentTemplateName *CheckQTN =
|
|
|
|
DependentTemplateNames.FindNodeOrInsertPos(ID, InsertPos);
|
|
|
|
assert(!CheckQTN && "Dependent type name canonicalization broken");
|
|
|
|
(void)CheckQTN;
|
2009-03-31 02:58:21 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
DependentTemplateNames.InsertNode(QTN, InsertPos);
|
|
|
|
return TemplateName(QTN);
|
|
|
|
}
|
|
|
|
|
2009-11-04 03:56:37 +03:00
|
|
|
/// \brief Retrieve the template name that represents a dependent
|
|
|
|
/// template name such as \c MetaFun::template operator+.
|
|
|
|
TemplateName
|
|
|
|
ASTContext::getDependentTemplateName(NestedNameSpecifier *NNS,
|
2011-01-12 12:06:06 +03:00
|
|
|
OverloadedOperatorKind Operator) const {
|
2009-11-04 03:56:37 +03:00
|
|
|
assert((!NNS || NNS->isDependent()) &&
|
|
|
|
"Nested name specifier must be dependent");
|
|
|
|
|
|
|
|
llvm::FoldingSetNodeID ID;
|
|
|
|
DependentTemplateName::Profile(ID, NNS, Operator);
|
|
|
|
|
|
|
|
void *InsertPos = 0;
|
Fix an obscure crash found in the Boost.MPL test suite, along with a
ton of potential crashes of the same kind. The fundamental problem is
that type creation was following a dangerous pattern when using its
FoldingSets:
1) Use FindNodeOrInsertPos to see if the type is available
2) If not, and we aren't looking at a canonical type, build the
canonical type
3) Build and insert the new node into the FoldingSet
The problem here is that building the canonical type can, in very rare
circumstances, force the hash table inside the FoldingSet to
reallocate. That invalidates the insertion position we computed in
step 1, and in step 3 we end up inserting the new node into the wrong
place. BOOM!
I've audited all of ASTContext, fixing this problem everywhere I found
it. The vast majority of wrong code was C++-specific (and *ahem*
written by me), so I also audited other major folding sets in the C++
code (e.g., template specializations), but found no other instances of
this problem.
git-svn-id: https://llvm.org/svn/llvm-project/cfe/trunk@95315 91177308-0d34-0410-b5e6-96231b3b80d8
2010-02-04 21:10:26 +03:00
|
|
|
DependentTemplateName *QTN
|
|
|
|
= DependentTemplateNames.FindNodeOrInsertPos(ID, InsertPos);
|
2009-11-04 03:56:37 +03:00
|
|
|
|
|
|
|
if (QTN)
|
|
|
|
return TemplateName(QTN);
|
|
|
|
|
|
|
|
NestedNameSpecifier *CanonNNS = getCanonicalNestedNameSpecifier(NNS);
|
|
|
|
if (CanonNNS == NNS) {
|
|
|
|
QTN = new (*this,4) DependentTemplateName(NNS, Operator);
|
|
|
|
} else {
|
|
|
|
TemplateName Canon = getDependentTemplateName(CanonNNS, Operator);
|
|
|
|
QTN = new (*this,4) DependentTemplateName(NNS, Operator, Canon);
|
Fix an obscure crash found in the Boost.MPL test suite, along with a
ton of potential crashes of the same kind. The fundamental problem is
that type creation was following a dangerous pattern when using its
FoldingSets:
1) Use FindNodeOrInsertPos to see if the type is available
2) If not, and we aren't looking at a canonical type, build the
canonical type
3) Build and insert the new node into the FoldingSet
The problem here is that building the canonical type can, in very rare
circumstances, force the hash table inside the FoldingSet to
reallocate. That invalidates the insertion position we computed in
step 1, and in step 3 we end up inserting the new node into the wrong
place. BOOM!
I've audited all of ASTContext, fixing this problem everywhere I found
it. The vast majority of wrong code was C++-specific (and *ahem*
written by me), so I also audited other major folding sets in the C++
code (e.g., template specializations), but found no other instances of
this problem.
git-svn-id: https://llvm.org/svn/llvm-project/cfe/trunk@95315 91177308-0d34-0410-b5e6-96231b3b80d8
2010-02-04 21:10:26 +03:00
|
|
|
|
|
|
|
DependentTemplateName *CheckQTN
|
|
|
|
= DependentTemplateNames.FindNodeOrInsertPos(ID, InsertPos);
|
|
|
|
assert(!CheckQTN && "Dependent template name canonicalization broken");
|
|
|
|
(void)CheckQTN;
|
2009-11-04 03:56:37 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
DependentTemplateNames.InsertNode(QTN, InsertPos);
|
|
|
|
return TemplateName(QTN);
|
|
|
|
}
|
|
|
|
|
2011-06-30 12:33:18 +04:00
|
|
|
TemplateName
|
|
|
|
ASTContext::getSubstTemplateTemplateParm(TemplateTemplateParmDecl *param,
|
|
|
|
TemplateName replacement) const {
|
|
|
|
llvm::FoldingSetNodeID ID;
|
|
|
|
SubstTemplateTemplateParmStorage::Profile(ID, param, replacement);
|
|
|
|
|
|
|
|
void *insertPos = 0;
|
|
|
|
SubstTemplateTemplateParmStorage *subst
|
|
|
|
= SubstTemplateTemplateParms.FindNodeOrInsertPos(ID, insertPos);
|
|
|
|
|
|
|
|
if (!subst) {
|
|
|
|
subst = new (*this) SubstTemplateTemplateParmStorage(param, replacement);
|
|
|
|
SubstTemplateTemplateParms.InsertNode(subst, insertPos);
|
|
|
|
}
|
|
|
|
|
|
|
|
return TemplateName(subst);
|
|
|
|
}
|
|
|
|
|
2011-01-15 09:45:20 +03:00
|
|
|
TemplateName
|
|
|
|
ASTContext::getSubstTemplateTemplateParmPack(TemplateTemplateParmDecl *Param,
|
|
|
|
const TemplateArgument &ArgPack) const {
|
|
|
|
ASTContext &Self = const_cast<ASTContext &>(*this);
|
|
|
|
llvm::FoldingSetNodeID ID;
|
|
|
|
SubstTemplateTemplateParmPackStorage::Profile(ID, Self, Param, ArgPack);
|
|
|
|
|
|
|
|
void *InsertPos = 0;
|
|
|
|
SubstTemplateTemplateParmPackStorage *Subst
|
|
|
|
= SubstTemplateTemplateParmPacks.FindNodeOrInsertPos(ID, InsertPos);
|
|
|
|
|
|
|
|
if (!Subst) {
|
2011-06-30 12:33:18 +04:00
|
|
|
Subst = new (*this) SubstTemplateTemplateParmPackStorage(Param,
|
2011-01-15 09:45:20 +03:00
|
|
|
ArgPack.pack_size(),
|
|
|
|
ArgPack.pack_begin());
|
|
|
|
SubstTemplateTemplateParmPacks.InsertNode(Subst, InsertPos);
|
|
|
|
}
|
|
|
|
|
|
|
|
return TemplateName(Subst);
|
|
|
|
}
|
|
|
|
|
2008-11-03 17:12:49 +03:00
|
|
|
/// getFromTargetType - Given one of the integer types provided by
|
2008-11-03 18:57:00 +03:00
|
|
|
/// TargetInfo, produce the corresponding type. The unsigned @p Type
|
|
|
|
/// is actually a value of type @c TargetInfo::IntType.
|
2009-10-24 03:03:21 +04:00
|
|
|
CanQualType ASTContext::getFromTargetType(unsigned Type) const {
|
2008-11-03 17:12:49 +03:00
|
|
|
switch (Type) {
|
2009-10-24 03:03:21 +04:00
|
|
|
case TargetInfo::NoInt: return CanQualType();
|
2008-11-03 17:12:49 +03:00
|
|
|
case TargetInfo::SignedShort: return ShortTy;
|
|
|
|
case TargetInfo::UnsignedShort: return UnsignedShortTy;
|
|
|
|
case TargetInfo::SignedInt: return IntTy;
|
|
|
|
case TargetInfo::UnsignedInt: return UnsignedIntTy;
|
|
|
|
case TargetInfo::SignedLong: return LongTy;
|
|
|
|
case TargetInfo::UnsignedLong: return UnsignedLongTy;
|
|
|
|
case TargetInfo::SignedLongLong: return LongLongTy;
|
|
|
|
case TargetInfo::UnsignedLongLong: return UnsignedLongLongTy;
|
|
|
|
}
|
|
|
|
|
2011-09-23 09:06:16 +04:00
|
|
|
llvm_unreachable("Unhandled TargetInfo::IntType value");
|
2008-11-03 17:12:49 +03:00
|
|
|
}
|
2008-07-25 03:58:27 +04:00
|
|
|
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
// Type Predicates.
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
|
2009-02-19 00:49:28 +03:00
|
|
|
/// getObjCGCAttr - Returns one of GCNone, Weak or Strong objc's
|
|
|
|
/// garbage collection attribute.
|
|
|
|
///
|
2011-01-12 03:34:59 +03:00
|
|
|
Qualifiers::GC ASTContext::getObjCGCAttrKind(QualType Ty) const {
|
2012-03-11 11:00:24 +04:00
|
|
|
if (getLangOpts().getGC() == LangOptions::NonGC)
|
2011-01-12 03:34:59 +03:00
|
|
|
return Qualifiers::GCNone;
|
|
|
|
|
2012-03-11 11:00:24 +04:00
|
|
|
assert(getLangOpts().ObjC1);
|
2011-01-12 03:34:59 +03:00
|
|
|
Qualifiers::GC GCAttrs = Ty.getObjCGCAttr();
|
|
|
|
|
|
|
|
// Default behaviour under objective-C's gc is for ObjC pointers
|
|
|
|
// (or pointers to them) be treated as though they were declared
|
|
|
|
// as __strong.
|
|
|
|
if (GCAttrs == Qualifiers::GCNone) {
|
|
|
|
if (Ty->isObjCObjectPointerType() || Ty->isBlockPointerType())
|
|
|
|
return Qualifiers::Strong;
|
|
|
|
else if (Ty->isPointerType())
|
|
|
|
return getObjCGCAttrKind(Ty->getAs<PointerType>()->getPointeeType());
|
|
|
|
} else {
|
|
|
|
// It's not valid to set GC attributes on anything that isn't a
|
|
|
|
// pointer.
|
|
|
|
#ifndef NDEBUG
|
|
|
|
QualType CT = Ty->getCanonicalTypeInternal();
|
|
|
|
while (const ArrayType *AT = dyn_cast<ArrayType>(CT))
|
|
|
|
CT = AT->getElementType();
|
|
|
|
assert(CT->isAnyPointerType() || CT->isBlockPointerType());
|
|
|
|
#endif
|
2009-02-19 00:49:28 +03:00
|
|
|
}
|
2009-02-19 01:53:11 +03:00
|
|
|
return GCAttrs;
|
2009-02-19 00:49:28 +03:00
|
|
|
}
|
|
|
|
|
2008-04-07 10:51:04 +04:00
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
// Type Compatibility Testing
|
|
|
|
//===----------------------------------------------------------------------===//
|
2007-11-01 08:03:41 +03:00
|
|
|
|
2009-09-09 19:08:12 +04:00
|
|
|
/// areCompatVectorTypes - Return true if the two specified vector types are
|
2008-04-07 10:51:04 +04:00
|
|
|
/// compatible.
|
|
|
|
static bool areCompatVectorTypes(const VectorType *LHS,
|
|
|
|
const VectorType *RHS) {
|
2009-10-23 00:10:53 +04:00
|
|
|
assert(LHS->isCanonicalUnqualified() && RHS->isCanonicalUnqualified());
|
2008-04-07 10:51:04 +04:00
|
|
|
return LHS->getElementType() == RHS->getElementType() &&
|
2008-10-05 21:34:18 +04:00
|
|
|
LHS->getNumElements() == RHS->getNumElements();
|
2008-04-07 10:51:04 +04:00
|
|
|
}
|
|
|
|
|
2010-08-06 14:14:59 +04:00
|
|
|
bool ASTContext::areCompatibleVectorTypes(QualType FirstVec,
|
|
|
|
QualType SecondVec) {
|
|
|
|
assert(FirstVec->isVectorType() && "FirstVec should be a vector type");
|
|
|
|
assert(SecondVec->isVectorType() && "SecondVec should be a vector type");
|
|
|
|
|
|
|
|
if (hasSameUnqualifiedType(FirstVec, SecondVec))
|
|
|
|
return true;
|
|
|
|
|
2010-11-12 20:24:54 +03:00
|
|
|
// Treat Neon vector types and most AltiVec vector types as if they are the
|
|
|
|
// equivalent GCC vector types.
|
2010-08-06 14:14:59 +04:00
|
|
|
const VectorType *First = FirstVec->getAs<VectorType>();
|
|
|
|
const VectorType *Second = SecondVec->getAs<VectorType>();
|
2010-11-12 20:24:54 +03:00
|
|
|
if (First->getNumElements() == Second->getNumElements() &&
|
2010-08-06 14:14:59 +04:00
|
|
|
hasSameType(First->getElementType(), Second->getElementType()) &&
|
2010-11-12 20:24:54 +03:00
|
|
|
First->getVectorKind() != VectorType::AltiVecPixel &&
|
|
|
|
First->getVectorKind() != VectorType::AltiVecBool &&
|
|
|
|
Second->getVectorKind() != VectorType::AltiVecPixel &&
|
|
|
|
Second->getVectorKind() != VectorType::AltiVecBool)
|
2010-08-06 14:14:59 +04:00
|
|
|
return true;
|
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2009-07-23 05:01:38 +04:00
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
// ObjCQualifiedIdTypesAreCompatible - Compatibility testing for qualified id's.
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
|
|
|
|
/// ProtocolCompatibleWithProtocol - return 'true' if 'lProto' is in the
|
|
|
|
/// inheritance hierarchy of 'rProto'.
|
2011-01-12 12:06:06 +03:00
|
|
|
bool
|
|
|
|
ASTContext::ProtocolCompatibleWithProtocol(ObjCProtocolDecl *lProto,
|
|
|
|
ObjCProtocolDecl *rProto) const {
|
2012-01-01 22:09:12 +04:00
|
|
|
if (declaresSameEntity(lProto, rProto))
|
2009-07-11 03:34:53 +04:00
|
|
|
return true;
|
2009-07-23 05:01:38 +04:00
|
|
|
for (ObjCProtocolDecl::protocol_iterator PI = rProto->protocol_begin(),
|
|
|
|
E = rProto->protocol_end(); PI != E; ++PI)
|
|
|
|
if (ProtocolCompatibleWithProtocol(lProto, *PI))
|
|
|
|
return true;
|
|
|
|
return false;
|
|
|
|
}
|
2009-07-11 03:34:53 +04:00
|
|
|
|
2009-07-23 05:01:38 +04:00
|
|
|
/// QualifiedIdConformsQualifiedId - compare id<p,...> with id<p1,...>
|
|
|
|
/// return true if lhs's protocols conform to rhs's protocol; false
|
|
|
|
/// otherwise.
|
|
|
|
bool ASTContext::QualifiedIdConformsQualifiedId(QualType lhs, QualType rhs) {
|
|
|
|
if (lhs->isObjCQualifiedIdType() && rhs->isObjCQualifiedIdType())
|
|
|
|
return ObjCQualifiedIdTypesAreCompatible(lhs, rhs, false);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2010-07-20 02:02:22 +04:00
|
|
|
/// ObjCQualifiedClassTypesAreCompatible - compare Class<p,...> and
|
|
|
|
/// Class<p1, ...>.
|
|
|
|
bool ASTContext::ObjCQualifiedClassTypesAreCompatible(QualType lhs,
|
|
|
|
QualType rhs) {
|
|
|
|
const ObjCObjectPointerType *lhsQID = lhs->getAs<ObjCObjectPointerType>();
|
|
|
|
const ObjCObjectPointerType *rhsOPT = rhs->getAs<ObjCObjectPointerType>();
|
|
|
|
assert ((lhsQID && rhsOPT) && "ObjCQualifiedClassTypesAreCompatible");
|
|
|
|
|
|
|
|
for (ObjCObjectPointerType::qual_iterator I = lhsQID->qual_begin(),
|
|
|
|
E = lhsQID->qual_end(); I != E; ++I) {
|
|
|
|
bool match = false;
|
|
|
|
ObjCProtocolDecl *lhsProto = *I;
|
|
|
|
for (ObjCObjectPointerType::qual_iterator J = rhsOPT->qual_begin(),
|
|
|
|
E = rhsOPT->qual_end(); J != E; ++J) {
|
|
|
|
ObjCProtocolDecl *rhsProto = *J;
|
|
|
|
if (ProtocolCompatibleWithProtocol(lhsProto, rhsProto)) {
|
|
|
|
match = true;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (!match)
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2009-07-23 05:01:38 +04:00
|
|
|
/// ObjCQualifiedIdTypesAreCompatible - We know that one of lhs/rhs is an
|
|
|
|
/// ObjCQualifiedIDType.
|
|
|
|
bool ASTContext::ObjCQualifiedIdTypesAreCompatible(QualType lhs, QualType rhs,
|
|
|
|
bool compare) {
|
|
|
|
// Allow id<P..> and an 'id' or void* type in all cases.
|
2009-09-09 19:08:12 +04:00
|
|
|
if (lhs->isVoidPointerType() ||
|
2009-07-23 05:01:38 +04:00
|
|
|
lhs->isObjCIdType() || lhs->isObjCClassType())
|
|
|
|
return true;
|
2009-09-09 19:08:12 +04:00
|
|
|
else if (rhs->isVoidPointerType() ||
|
2009-07-23 05:01:38 +04:00
|
|
|
rhs->isObjCIdType() || rhs->isObjCClassType())
|
|
|
|
return true;
|
|
|
|
|
|
|
|
if (const ObjCObjectPointerType *lhsQID = lhs->getAsObjCQualifiedIdType()) {
|
2009-09-22 03:43:11 +04:00
|
|
|
const ObjCObjectPointerType *rhsOPT = rhs->getAs<ObjCObjectPointerType>();
|
2009-09-09 19:08:12 +04:00
|
|
|
|
2009-07-23 05:01:38 +04:00
|
|
|
if (!rhsOPT) return false;
|
2009-09-09 19:08:12 +04:00
|
|
|
|
2009-07-23 05:01:38 +04:00
|
|
|
if (rhsOPT->qual_empty()) {
|
2009-09-09 19:08:12 +04:00
|
|
|
// If the RHS is a unqualified interface pointer "NSString*",
|
2009-07-23 05:01:38 +04:00
|
|
|
// make sure we check the class hierarchy.
|
|
|
|
if (ObjCInterfaceDecl *rhsID = rhsOPT->getInterfaceDecl()) {
|
|
|
|
for (ObjCObjectPointerType::qual_iterator I = lhsQID->qual_begin(),
|
|
|
|
E = lhsQID->qual_end(); I != E; ++I) {
|
|
|
|
// when comparing an id<P> on lhs with a static type on rhs,
|
|
|
|
// see if static class implements all of id's protocols, directly or
|
|
|
|
// through its super class and categories.
|
2009-08-12 02:02:25 +04:00
|
|
|
if (!rhsID->ClassImplementsProtocol(*I, true))
|
2009-07-23 05:01:38 +04:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
// If there are no qualifiers and no interface, we have an 'id'.
|
|
|
|
return true;
|
|
|
|
}
|
2009-09-09 19:08:12 +04:00
|
|
|
// Both the right and left sides have qualifiers.
|
2009-07-23 05:01:38 +04:00
|
|
|
for (ObjCObjectPointerType::qual_iterator I = lhsQID->qual_begin(),
|
|
|
|
E = lhsQID->qual_end(); I != E; ++I) {
|
|
|
|
ObjCProtocolDecl *lhsProto = *I;
|
|
|
|
bool match = false;
|
2009-07-15 22:40:39 +04:00
|
|
|
|
|
|
|
// when comparing an id<P> on lhs with a static type on rhs,
|
|
|
|
// see if static class implements all of id's protocols, directly or
|
|
|
|
// through its super class and categories.
|
2009-07-23 05:01:38 +04:00
|
|
|
for (ObjCObjectPointerType::qual_iterator J = rhsOPT->qual_begin(),
|
|
|
|
E = rhsOPT->qual_end(); J != E; ++J) {
|
|
|
|
ObjCProtocolDecl *rhsProto = *J;
|
|
|
|
if (ProtocolCompatibleWithProtocol(lhsProto, rhsProto) ||
|
|
|
|
(compare && ProtocolCompatibleWithProtocol(rhsProto, lhsProto))) {
|
|
|
|
match = true;
|
2009-07-16 20:21:02 +04:00
|
|
|
break;
|
|
|
|
}
|
2009-07-15 22:40:39 +04:00
|
|
|
}
|
2009-09-09 19:08:12 +04:00
|
|
|
// If the RHS is a qualified interface pointer "NSString<P>*",
|
2009-07-23 05:01:38 +04:00
|
|
|
// make sure we check the class hierarchy.
|
|
|
|
if (ObjCInterfaceDecl *rhsID = rhsOPT->getInterfaceDecl()) {
|
|
|
|
for (ObjCObjectPointerType::qual_iterator I = lhsQID->qual_begin(),
|
|
|
|
E = lhsQID->qual_end(); I != E; ++I) {
|
|
|
|
// when comparing an id<P> on lhs with a static type on rhs,
|
|
|
|
// see if static class implements all of id's protocols, directly or
|
|
|
|
// through its super class and categories.
|
2009-08-12 02:02:25 +04:00
|
|
|
if (rhsID->ClassImplementsProtocol(*I, true)) {
|
2009-07-23 05:01:38 +04:00
|
|
|
match = true;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (!match)
|
2009-07-15 22:40:39 +04:00
|
|
|
return false;
|
|
|
|
}
|
2009-09-09 19:08:12 +04:00
|
|
|
|
2009-07-15 22:40:39 +04:00
|
|
|
return true;
|
|
|
|
}
|
2009-09-09 19:08:12 +04:00
|
|
|
|
2009-07-23 05:01:38 +04:00
|
|
|
const ObjCObjectPointerType *rhsQID = rhs->getAsObjCQualifiedIdType();
|
|
|
|
assert(rhsQID && "One of the LHS/RHS should be id<x>");
|
|
|
|
|
2009-09-09 19:08:12 +04:00
|
|
|
if (const ObjCObjectPointerType *lhsOPT =
|
2009-07-23 05:01:38 +04:00
|
|
|
lhs->getAsObjCInterfacePointerType()) {
|
2010-11-01 23:47:16 +03:00
|
|
|
// If both the right and left sides have qualifiers.
|
2009-07-23 05:01:38 +04:00
|
|
|
for (ObjCObjectPointerType::qual_iterator I = lhsOPT->qual_begin(),
|
|
|
|
E = lhsOPT->qual_end(); I != E; ++I) {
|
|
|
|
ObjCProtocolDecl *lhsProto = *I;
|
|
|
|
bool match = false;
|
|
|
|
|
2010-11-01 23:47:16 +03:00
|
|
|
// when comparing an id<P> on rhs with a static type on lhs,
|
2009-07-23 05:01:38 +04:00
|
|
|
// see if static class implements all of id's protocols, directly or
|
|
|
|
// through its super class and categories.
|
2010-11-01 23:47:16 +03:00
|
|
|
// First, lhs protocols in the qualifier list must be found, direct
|
|
|
|
// or indirect in rhs's qualifier list or it is a mismatch.
|
2009-07-23 05:01:38 +04:00
|
|
|
for (ObjCObjectPointerType::qual_iterator J = rhsQID->qual_begin(),
|
|
|
|
E = rhsQID->qual_end(); J != E; ++J) {
|
|
|
|
ObjCProtocolDecl *rhsProto = *J;
|
|
|
|
if (ProtocolCompatibleWithProtocol(lhsProto, rhsProto) ||
|
|
|
|
(compare && ProtocolCompatibleWithProtocol(rhsProto, lhsProto))) {
|
|
|
|
match = true;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (!match)
|
|
|
|
return false;
|
|
|
|
}
|
2010-11-01 23:47:16 +03:00
|
|
|
|
|
|
|
// Static class's protocols, or its super class or category protocols
|
|
|
|
// must be found, direct or indirect in rhs's qualifier list or it is a mismatch.
|
|
|
|
if (ObjCInterfaceDecl *lhsID = lhsOPT->getInterfaceDecl()) {
|
|
|
|
llvm::SmallPtrSet<ObjCProtocolDecl *, 8> LHSInheritedProtocols;
|
|
|
|
CollectInheritedProtocols(lhsID, LHSInheritedProtocols);
|
|
|
|
// This is rather dubious but matches gcc's behavior. If lhs has
|
|
|
|
// no type qualifier and its class has no static protocol(s)
|
|
|
|
// assume that it is mismatch.
|
|
|
|
if (LHSInheritedProtocols.empty() && lhsOPT->qual_empty())
|
|
|
|
return false;
|
|
|
|
for (llvm::SmallPtrSet<ObjCProtocolDecl*,8>::iterator I =
|
|
|
|
LHSInheritedProtocols.begin(),
|
|
|
|
E = LHSInheritedProtocols.end(); I != E; ++I) {
|
|
|
|
bool match = false;
|
|
|
|
ObjCProtocolDecl *lhsProto = (*I);
|
|
|
|
for (ObjCObjectPointerType::qual_iterator J = rhsQID->qual_begin(),
|
|
|
|
E = rhsQID->qual_end(); J != E; ++J) {
|
|
|
|
ObjCProtocolDecl *rhsProto = *J;
|
|
|
|
if (ProtocolCompatibleWithProtocol(lhsProto, rhsProto) ||
|
|
|
|
(compare && ProtocolCompatibleWithProtocol(rhsProto, lhsProto))) {
|
|
|
|
match = true;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (!match)
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
2009-07-23 05:01:38 +04:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
/// canAssignObjCInterfaces - Return true if the two interface types are
|
|
|
|
/// compatible for assignment from RHS to LHS. This handles validation of any
|
|
|
|
/// protocol qualifiers on the LHS or RHS.
|
|
|
|
///
|
|
|
|
bool ASTContext::canAssignObjCInterfaces(const ObjCObjectPointerType *LHSOPT,
|
|
|
|
const ObjCObjectPointerType *RHSOPT) {
|
2010-05-15 15:32:37 +04:00
|
|
|
const ObjCObjectType* LHS = LHSOPT->getObjectType();
|
|
|
|
const ObjCObjectType* RHS = RHSOPT->getObjectType();
|
|
|
|
|
2009-07-23 05:01:38 +04:00
|
|
|
// If either type represents the built-in 'id' or 'Class' types, return true.
|
2010-05-15 15:32:37 +04:00
|
|
|
if (LHS->isObjCUnqualifiedIdOrClass() ||
|
|
|
|
RHS->isObjCUnqualifiedIdOrClass())
|
2009-07-23 05:01:38 +04:00
|
|
|
return true;
|
|
|
|
|
2010-05-15 15:32:37 +04:00
|
|
|
if (LHS->isObjCQualifiedId() || RHS->isObjCQualifiedId())
|
2009-09-09 19:08:12 +04:00
|
|
|
return ObjCQualifiedIdTypesAreCompatible(QualType(LHSOPT,0),
|
|
|
|
QualType(RHSOPT,0),
|
2009-07-23 05:01:38 +04:00
|
|
|
false);
|
2010-07-20 02:02:22 +04:00
|
|
|
|
|
|
|
if (LHS->isObjCQualifiedClass() && RHS->isObjCQualifiedClass())
|
|
|
|
return ObjCQualifiedClassTypesAreCompatible(QualType(LHSOPT,0),
|
|
|
|
QualType(RHSOPT,0));
|
|
|
|
|
2010-05-15 15:32:37 +04:00
|
|
|
// If we have 2 user-defined types, fall into that path.
|
|
|
|
if (LHS->getInterface() && RHS->getInterface())
|
2009-07-23 05:01:38 +04:00
|
|
|
return canAssignObjCInterfaces(LHS, RHS);
|
2009-09-09 19:08:12 +04:00
|
|
|
|
2009-07-23 05:01:38 +04:00
|
|
|
return false;
|
2009-07-11 03:34:53 +04:00
|
|
|
}
|
|
|
|
|
2010-03-17 03:20:01 +03:00
|
|
|
/// canAssignObjCInterfacesInBlockPointer - This routine is specifically written
|
2011-04-15 09:22:18 +04:00
|
|
|
/// for providing type-safety for objective-c pointers used to pass/return
|
2010-03-17 03:20:01 +03:00
|
|
|
/// arguments in block literals. When passed as arguments, passing 'A*' where
|
|
|
|
/// 'id' is expected is not OK. Passing 'Sub *" where 'Super *" is expected is
|
|
|
|
/// not OK. For the return type, the opposite is not OK.
|
|
|
|
bool ASTContext::canAssignObjCInterfacesInBlockPointer(
|
|
|
|
const ObjCObjectPointerType *LHSOPT,
|
2011-03-14 19:07:00 +03:00
|
|
|
const ObjCObjectPointerType *RHSOPT,
|
|
|
|
bool BlockReturnType) {
|
2010-04-06 21:23:39 +04:00
|
|
|
if (RHSOPT->isObjCBuiltinType() || LHSOPT->isObjCIdType())
|
2010-03-17 03:20:01 +03:00
|
|
|
return true;
|
|
|
|
|
|
|
|
if (LHSOPT->isObjCBuiltinType()) {
|
|
|
|
return RHSOPT->isObjCBuiltinType() || RHSOPT->isObjCQualifiedIdType();
|
|
|
|
}
|
|
|
|
|
2010-04-06 21:23:39 +04:00
|
|
|
if (LHSOPT->isObjCQualifiedIdType() || RHSOPT->isObjCQualifiedIdType())
|
2010-03-17 03:20:01 +03:00
|
|
|
return ObjCQualifiedIdTypesAreCompatible(QualType(LHSOPT,0),
|
|
|
|
QualType(RHSOPT,0),
|
|
|
|
false);
|
|
|
|
|
|
|
|
const ObjCInterfaceType* LHS = LHSOPT->getInterfaceType();
|
|
|
|
const ObjCInterfaceType* RHS = RHSOPT->getInterfaceType();
|
|
|
|
if (LHS && RHS) { // We have 2 user-defined types.
|
|
|
|
if (LHS != RHS) {
|
|
|
|
if (LHS->getDecl()->isSuperClassOf(RHS->getDecl()))
|
2011-03-14 19:07:00 +03:00
|
|
|
return BlockReturnType;
|
2010-03-17 03:20:01 +03:00
|
|
|
if (RHS->getDecl()->isSuperClassOf(LHS->getDecl()))
|
2011-03-14 19:07:00 +03:00
|
|
|
return !BlockReturnType;
|
2010-03-17 03:20:01 +03:00
|
|
|
}
|
|
|
|
else
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2009-10-30 04:13:23 +03:00
|
|
|
/// getIntersectionOfProtocols - This routine finds the intersection of set
|
|
|
|
/// of protocols inherited from two distinct objective-c pointer objects.
|
|
|
|
/// It is used to build composite qualifier list of the composite type of
|
|
|
|
/// the conditional expression involving two objective-c pointer objects.
|
|
|
|
static
|
|
|
|
void getIntersectionOfProtocols(ASTContext &Context,
|
|
|
|
const ObjCObjectPointerType *LHSOPT,
|
|
|
|
const ObjCObjectPointerType *RHSOPT,
|
2011-07-23 14:55:15 +04:00
|
|
|
SmallVectorImpl<ObjCProtocolDecl *> &IntersectionOfProtocols) {
|
2009-10-30 04:13:23 +03:00
|
|
|
|
2010-05-15 15:32:37 +04:00
|
|
|
const ObjCObjectType* LHS = LHSOPT->getObjectType();
|
|
|
|
const ObjCObjectType* RHS = RHSOPT->getObjectType();
|
|
|
|
assert(LHS->getInterface() && "LHS must have an interface base");
|
|
|
|
assert(RHS->getInterface() && "RHS must have an interface base");
|
2009-10-30 04:13:23 +03:00
|
|
|
|
|
|
|
llvm::SmallPtrSet<ObjCProtocolDecl *, 8> InheritedProtocolSet;
|
|
|
|
unsigned LHSNumProtocols = LHS->getNumProtocols();
|
|
|
|
if (LHSNumProtocols > 0)
|
|
|
|
InheritedProtocolSet.insert(LHS->qual_begin(), LHS->qual_end());
|
|
|
|
else {
|
2010-02-12 22:27:33 +03:00
|
|
|
llvm::SmallPtrSet<ObjCProtocolDecl *, 8> LHSInheritedProtocols;
|
2010-05-15 15:32:37 +04:00
|
|
|
Context.CollectInheritedProtocols(LHS->getInterface(),
|
|
|
|
LHSInheritedProtocols);
|
2009-10-30 04:13:23 +03:00
|
|
|
InheritedProtocolSet.insert(LHSInheritedProtocols.begin(),
|
|
|
|
LHSInheritedProtocols.end());
|
|
|
|
}
|
|
|
|
|
|
|
|
unsigned RHSNumProtocols = RHS->getNumProtocols();
|
|
|
|
if (RHSNumProtocols > 0) {
|
2010-04-19 20:39:44 +04:00
|
|
|
ObjCProtocolDecl **RHSProtocols =
|
|
|
|
const_cast<ObjCProtocolDecl **>(RHS->qual_begin());
|
2009-10-30 04:13:23 +03:00
|
|
|
for (unsigned i = 0; i < RHSNumProtocols; ++i)
|
|
|
|
if (InheritedProtocolSet.count(RHSProtocols[i]))
|
|
|
|
IntersectionOfProtocols.push_back(RHSProtocols[i]);
|
2011-08-18 03:08:45 +04:00
|
|
|
} else {
|
2010-02-12 22:27:33 +03:00
|
|
|
llvm::SmallPtrSet<ObjCProtocolDecl *, 8> RHSInheritedProtocols;
|
2010-05-15 15:32:37 +04:00
|
|
|
Context.CollectInheritedProtocols(RHS->getInterface(),
|
|
|
|
RHSInheritedProtocols);
|
2010-02-12 22:27:33 +03:00
|
|
|
for (llvm::SmallPtrSet<ObjCProtocolDecl*,8>::iterator I =
|
|
|
|
RHSInheritedProtocols.begin(),
|
|
|
|
E = RHSInheritedProtocols.end(); I != E; ++I)
|
|
|
|
if (InheritedProtocolSet.count((*I)))
|
|
|
|
IntersectionOfProtocols.push_back((*I));
|
2009-10-30 04:13:23 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2009-10-28 02:02:38 +03:00
|
|
|
/// areCommonBaseCompatible - Returns common base class of the two classes if
|
|
|
|
/// one found. Note that this is O'2 algorithm. But it will be called as the
|
|
|
|
/// last type comparison in a ?-exp of ObjC pointer types before a
|
|
|
|
/// warning is issued. So, its invokation is extremely rare.
|
|
|
|
QualType ASTContext::areCommonBaseCompatible(
|
2010-05-15 15:32:37 +04:00
|
|
|
const ObjCObjectPointerType *Lptr,
|
|
|
|
const ObjCObjectPointerType *Rptr) {
|
|
|
|
const ObjCObjectType *LHS = Lptr->getObjectType();
|
|
|
|
const ObjCObjectType *RHS = Rptr->getObjectType();
|
|
|
|
const ObjCInterfaceDecl* LDecl = LHS->getInterface();
|
|
|
|
const ObjCInterfaceDecl* RDecl = RHS->getInterface();
|
2011-12-15 04:29:59 +04:00
|
|
|
if (!LDecl || !RDecl || (declaresSameEntity(LDecl, RDecl)))
|
2009-10-28 02:02:38 +03:00
|
|
|
return QualType();
|
|
|
|
|
2011-04-19 01:16:59 +04:00
|
|
|
do {
|
2010-05-15 15:32:37 +04:00
|
|
|
LHS = cast<ObjCInterfaceType>(getObjCInterfaceType(LDecl));
|
2009-10-30 04:13:23 +03:00
|
|
|
if (canAssignObjCInterfaces(LHS, RHS)) {
|
2011-07-23 14:55:15 +04:00
|
|
|
SmallVector<ObjCProtocolDecl *, 8> Protocols;
|
2010-05-15 15:32:37 +04:00
|
|
|
getIntersectionOfProtocols(*this, Lptr, Rptr, Protocols);
|
|
|
|
|
|
|
|
QualType Result = QualType(LHS, 0);
|
|
|
|
if (!Protocols.empty())
|
|
|
|
Result = getObjCObjectType(Result, Protocols.data(), Protocols.size());
|
|
|
|
Result = getObjCObjectPointerType(Result);
|
|
|
|
return Result;
|
2009-10-30 04:13:23 +03:00
|
|
|
}
|
2011-04-19 01:16:59 +04:00
|
|
|
} while ((LDecl = LDecl->getSuperClass()));
|
2009-10-28 02:02:38 +03:00
|
|
|
|
|
|
|
return QualType();
|
|
|
|
}
|
|
|
|
|
2010-05-15 15:32:37 +04:00
|
|
|
bool ASTContext::canAssignObjCInterfaces(const ObjCObjectType *LHS,
|
|
|
|
const ObjCObjectType *RHS) {
|
|
|
|
assert(LHS->getInterface() && "LHS is not an interface type");
|
|
|
|
assert(RHS->getInterface() && "RHS is not an interface type");
|
|
|
|
|
2008-04-07 10:51:04 +04:00
|
|
|
// Verify that the base decls are compatible: the RHS must be a subclass of
|
|
|
|
// the LHS.
|
2010-05-15 15:32:37 +04:00
|
|
|
if (!LHS->getInterface()->isSuperClassOf(RHS->getInterface()))
|
2008-04-07 10:51:04 +04:00
|
|
|
return false;
|
2009-09-09 19:08:12 +04:00
|
|
|
|
2008-04-07 10:51:04 +04:00
|
|
|
// RHS must have a superset of the protocols in the LHS. If the LHS is not
|
|
|
|
// protocol qualified at all, then we are good.
|
2009-07-18 19:33:26 +04:00
|
|
|
if (LHS->getNumProtocols() == 0)
|
2008-04-07 10:51:04 +04:00
|
|
|
return true;
|
2009-09-09 19:08:12 +04:00
|
|
|
|
2011-04-08 22:25:29 +04:00
|
|
|
// Okay, we know the LHS has protocol qualifiers. If the RHS doesn't,
|
|
|
|
// more detailed analysis is required.
|
|
|
|
if (RHS->getNumProtocols() == 0) {
|
|
|
|
// OK, if LHS is a superclass of RHS *and*
|
|
|
|
// this superclass is assignment compatible with LHS.
|
|
|
|
// false otherwise.
|
2011-04-12 20:34:14 +04:00
|
|
|
bool IsSuperClass =
|
|
|
|
LHS->getInterface()->isSuperClassOf(RHS->getInterface());
|
|
|
|
if (IsSuperClass) {
|
2011-04-08 22:25:29 +04:00
|
|
|
// OK if conversion of LHS to SuperClass results in narrowing of types
|
|
|
|
// ; i.e., SuperClass may implement at least one of the protocols
|
|
|
|
// in LHS's protocol list. Example, SuperObj<P1> = lhs<P1,P2> is ok.
|
|
|
|
// But not SuperObj<P1,P2,P3> = lhs<P1,P2>.
|
|
|
|
llvm::SmallPtrSet<ObjCProtocolDecl *, 8> SuperClassInheritedProtocols;
|
2011-04-12 20:34:14 +04:00
|
|
|
CollectInheritedProtocols(RHS->getInterface(), SuperClassInheritedProtocols);
|
2011-04-08 22:25:29 +04:00
|
|
|
// If super class has no protocols, it is not a match.
|
|
|
|
if (SuperClassInheritedProtocols.empty())
|
|
|
|
return false;
|
|
|
|
|
|
|
|
for (ObjCObjectType::qual_iterator LHSPI = LHS->qual_begin(),
|
|
|
|
LHSPE = LHS->qual_end();
|
|
|
|
LHSPI != LHSPE; LHSPI++) {
|
|
|
|
bool SuperImplementsProtocol = false;
|
|
|
|
ObjCProtocolDecl *LHSProto = (*LHSPI);
|
|
|
|
|
|
|
|
for (llvm::SmallPtrSet<ObjCProtocolDecl*,8>::iterator I =
|
|
|
|
SuperClassInheritedProtocols.begin(),
|
|
|
|
E = SuperClassInheritedProtocols.end(); I != E; ++I) {
|
|
|
|
ObjCProtocolDecl *SuperClassProto = (*I);
|
|
|
|
if (SuperClassProto->lookupProtocolNamed(LHSProto->getIdentifier())) {
|
|
|
|
SuperImplementsProtocol = true;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (!SuperImplementsProtocol)
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
2009-09-09 19:08:12 +04:00
|
|
|
|
2010-05-15 15:32:37 +04:00
|
|
|
for (ObjCObjectType::qual_iterator LHSPI = LHS->qual_begin(),
|
|
|
|
LHSPE = LHS->qual_end();
|
2009-03-01 19:12:44 +03:00
|
|
|
LHSPI != LHSPE; LHSPI++) {
|
|
|
|
bool RHSImplementsProtocol = false;
|
|
|
|
|
|
|
|
// If the RHS doesn't implement the protocol on the left, the types
|
|
|
|
// are incompatible.
|
2010-05-15 15:32:37 +04:00
|
|
|
for (ObjCObjectType::qual_iterator RHSPI = RHS->qual_begin(),
|
|
|
|
RHSPE = RHS->qual_end();
|
2009-07-16 20:21:02 +04:00
|
|
|
RHSPI != RHSPE; RHSPI++) {
|
|
|
|
if ((*RHSPI)->lookupProtocolNamed((*LHSPI)->getIdentifier())) {
|
2009-03-01 19:12:44 +03:00
|
|
|
RHSImplementsProtocol = true;
|
2009-07-16 20:21:02 +04:00
|
|
|
break;
|
|
|
|
}
|
2009-03-01 19:12:44 +03:00
|
|
|
}
|
|
|
|
// FIXME: For better diagnostics, consider passing back the protocol name.
|
|
|
|
if (!RHSImplementsProtocol)
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
// The RHS implements all protocols listed on the LHS.
|
|
|
|
return true;
|
2008-04-07 10:51:04 +04:00
|
|
|
}
|
|
|
|
|
2009-02-12 20:52:19 +03:00
|
|
|
bool ASTContext::areComparableObjCPointerTypes(QualType LHS, QualType RHS) {
|
|
|
|
// get the "pointed to" types
|
2009-09-22 03:43:11 +04:00
|
|
|
const ObjCObjectPointerType *LHSOPT = LHS->getAs<ObjCObjectPointerType>();
|
|
|
|
const ObjCObjectPointerType *RHSOPT = RHS->getAs<ObjCObjectPointerType>();
|
2009-09-09 19:08:12 +04:00
|
|
|
|
2009-07-11 03:34:53 +04:00
|
|
|
if (!LHSOPT || !RHSOPT)
|
2009-02-12 20:52:19 +03:00
|
|
|
return false;
|
2009-07-11 03:34:53 +04:00
|
|
|
|
|
|
|
return canAssignObjCInterfaces(LHSOPT, RHSOPT) ||
|
|
|
|
canAssignObjCInterfaces(RHSOPT, LHSOPT);
|
2009-02-12 20:52:19 +03:00
|
|
|
}
|
|
|
|
|
2010-08-07 15:51:51 +04:00
|
|
|
bool ASTContext::canBindObjCObjectType(QualType To, QualType From) {
|
|
|
|
return canAssignObjCInterfaces(
|
|
|
|
getObjCObjectPointerType(To)->getAs<ObjCObjectPointerType>(),
|
|
|
|
getObjCObjectPointerType(From)->getAs<ObjCObjectPointerType>());
|
|
|
|
}
|
|
|
|
|
2009-09-09 19:08:12 +04:00
|
|
|
/// typesAreCompatible - C99 6.7.3p9: For two qualified types to be compatible,
|
2007-10-16 00:41:53 +04:00
|
|
|
/// both shall have the identically qualified version of a compatible type.
|
2009-09-09 19:08:12 +04:00
|
|
|
/// C99 6.2.7p1: Two types have compatible types if their types are the
|
2007-10-16 00:41:53 +04:00
|
|
|
/// same. See 6.7.[2,3,5] for additional rules.
|
2010-07-29 19:18:02 +04:00
|
|
|
bool ASTContext::typesAreCompatible(QualType LHS, QualType RHS,
|
|
|
|
bool CompareUnqualified) {
|
2012-03-11 11:00:24 +04:00
|
|
|
if (getLangOpts().CPlusPlus)
|
2010-02-04 00:02:30 +03:00
|
|
|
return hasSameType(LHS, RHS);
|
|
|
|
|
2010-07-29 19:18:02 +04:00
|
|
|
return !mergeTypes(LHS, RHS, false, CompareUnqualified).isNull();
|
2008-08-22 04:56:42 +04:00
|
|
|
}
|
|
|
|
|
2011-07-13 02:05:16 +04:00
|
|
|
bool ASTContext::propertyTypesAreCompatible(QualType LHS, QualType RHS) {
|
2011-07-13 03:20:13 +04:00
|
|
|
return typesAreCompatible(LHS, RHS);
|
2011-07-13 02:05:16 +04:00
|
|
|
}
|
|
|
|
|
2010-03-17 03:20:01 +03:00
|
|
|
bool ASTContext::typesAreBlockPointerCompatible(QualType LHS, QualType RHS) {
|
|
|
|
return !mergeTypes(LHS, RHS, true).isNull();
|
|
|
|
}
|
|
|
|
|
2010-10-24 22:30:18 +04:00
|
|
|
/// mergeTransparentUnionType - if T is a transparent union type and a member
|
|
|
|
/// of T is compatible with SubType, return the merged type, else return
|
|
|
|
/// QualType()
|
|
|
|
QualType ASTContext::mergeTransparentUnionType(QualType T, QualType SubType,
|
|
|
|
bool OfBlockPointer,
|
|
|
|
bool Unqualified) {
|
|
|
|
if (const RecordType *UT = T->getAsUnionType()) {
|
|
|
|
RecordDecl *UD = UT->getDecl();
|
|
|
|
if (UD->hasAttr<TransparentUnionAttr>()) {
|
|
|
|
for (RecordDecl::field_iterator it = UD->field_begin(),
|
|
|
|
itend = UD->field_end(); it != itend; ++it) {
|
2010-12-03 00:00:06 +03:00
|
|
|
QualType ET = it->getType().getUnqualifiedType();
|
2010-10-24 22:30:18 +04:00
|
|
|
QualType MT = mergeTypes(ET, SubType, OfBlockPointer, Unqualified);
|
|
|
|
if (!MT.isNull())
|
|
|
|
return MT;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return QualType();
|
|
|
|
}
|
|
|
|
|
|
|
|
/// mergeFunctionArgumentTypes - merge two types which appear as function
|
|
|
|
/// argument types
|
|
|
|
QualType ASTContext::mergeFunctionArgumentTypes(QualType lhs, QualType rhs,
|
|
|
|
bool OfBlockPointer,
|
|
|
|
bool Unqualified) {
|
|
|
|
// GNU extension: two types are compatible if they appear as a function
|
|
|
|
// argument, one of the types is a transparent union type and the other
|
|
|
|
// type is compatible with a union member
|
|
|
|
QualType lmerge = mergeTransparentUnionType(lhs, rhs, OfBlockPointer,
|
|
|
|
Unqualified);
|
|
|
|
if (!lmerge.isNull())
|
|
|
|
return lmerge;
|
|
|
|
|
|
|
|
QualType rmerge = mergeTransparentUnionType(rhs, lhs, OfBlockPointer,
|
|
|
|
Unqualified);
|
|
|
|
if (!rmerge.isNull())
|
|
|
|
return rmerge;
|
|
|
|
|
|
|
|
return mergeTypes(lhs, rhs, OfBlockPointer, Unqualified);
|
|
|
|
}
|
|
|
|
|
2010-03-17 03:20:01 +03:00
|
|
|
QualType ASTContext::mergeFunctionTypes(QualType lhs, QualType rhs,
|
2010-07-29 19:18:02 +04:00
|
|
|
bool OfBlockPointer,
|
|
|
|
bool Unqualified) {
|
2009-09-22 03:43:11 +04:00
|
|
|
const FunctionType *lbase = lhs->getAs<FunctionType>();
|
|
|
|
const FunctionType *rbase = rhs->getAs<FunctionType>();
|
2009-02-27 02:50:07 +03:00
|
|
|
const FunctionProtoType *lproto = dyn_cast<FunctionProtoType>(lbase);
|
|
|
|
const FunctionProtoType *rproto = dyn_cast<FunctionProtoType>(rbase);
|
2008-08-22 04:56:42 +04:00
|
|
|
bool allLTypes = true;
|
|
|
|
bool allRTypes = true;
|
|
|
|
|
|
|
|
// Check return type
|
2010-03-17 03:20:01 +03:00
|
|
|
QualType retType;
|
2011-02-11 21:46:17 +03:00
|
|
|
if (OfBlockPointer) {
|
|
|
|
QualType RHS = rbase->getResultType();
|
|
|
|
QualType LHS = lbase->getResultType();
|
|
|
|
bool UnqualifiedResult = Unqualified;
|
|
|
|
if (!UnqualifiedResult)
|
|
|
|
UnqualifiedResult = (!RHS.hasQualifiers() && LHS.hasQualifiers());
|
2011-03-14 19:07:00 +03:00
|
|
|
retType = mergeTypes(LHS, RHS, true, UnqualifiedResult, true);
|
2011-02-11 21:46:17 +03:00
|
|
|
}
|
2010-03-17 03:20:01 +03:00
|
|
|
else
|
2010-12-15 04:06:38 +03:00
|
|
|
retType = mergeTypes(lbase->getResultType(), rbase->getResultType(), false,
|
|
|
|
Unqualified);
|
2008-08-22 04:56:42 +04:00
|
|
|
if (retType.isNull()) return QualType();
|
2010-07-29 19:18:02 +04:00
|
|
|
|
|
|
|
if (Unqualified)
|
|
|
|
retType = retType.getUnqualifiedType();
|
|
|
|
|
|
|
|
CanQualType LRetType = getCanonicalType(lbase->getResultType());
|
|
|
|
CanQualType RRetType = getCanonicalType(rbase->getResultType());
|
|
|
|
if (Unqualified) {
|
|
|
|
LRetType = LRetType.getUnqualifiedType();
|
|
|
|
RRetType = RRetType.getUnqualifiedType();
|
|
|
|
}
|
|
|
|
|
|
|
|
if (getCanonicalType(retType) != LRetType)
|
2008-10-05 21:34:18 +04:00
|
|
|
allLTypes = false;
|
2010-07-29 19:18:02 +04:00
|
|
|
if (getCanonicalType(retType) != RRetType)
|
2008-10-05 21:34:18 +04:00
|
|
|
allRTypes = false;
|
2010-12-15 04:06:38 +03:00
|
|
|
|
2010-04-28 20:20:58 +04:00
|
|
|
// FIXME: double check this
|
|
|
|
// FIXME: should we error if lbase->getRegParmAttr() != 0 &&
|
|
|
|
// rbase->getRegParmAttr() != 0 &&
|
|
|
|
// lbase->getRegParmAttr() != rbase->getRegParmAttr()?
|
2010-03-31 00:24:48 +04:00
|
|
|
FunctionType::ExtInfo lbaseInfo = lbase->getExtInfo();
|
|
|
|
FunctionType::ExtInfo rbaseInfo = rbase->getExtInfo();
|
2010-12-15 04:06:38 +03:00
|
|
|
|
|
|
|
// Compatible functions must have compatible calling conventions
|
|
|
|
if (!isSameCallConv(lbaseInfo.getCC(), rbaseInfo.getCC()))
|
|
|
|
return QualType();
|
|
|
|
|
|
|
|
// Regparm is part of the calling convention.
|
2011-04-09 12:18:08 +04:00
|
|
|
if (lbaseInfo.getHasRegParm() != rbaseInfo.getHasRegParm())
|
|
|
|
return QualType();
|
2010-12-15 04:06:38 +03:00
|
|
|
if (lbaseInfo.getRegParm() != rbaseInfo.getRegParm())
|
|
|
|
return QualType();
|
|
|
|
|
2011-06-16 03:02:42 +04:00
|
|
|
if (lbaseInfo.getProducesResult() != rbaseInfo.getProducesResult())
|
|
|
|
return QualType();
|
|
|
|
|
2011-10-05 04:05:34 +04:00
|
|
|
// functypes which return are preferred over those that do not.
|
|
|
|
if (lbaseInfo.getNoReturn() && !rbaseInfo.getNoReturn())
|
2010-04-28 20:20:58 +04:00
|
|
|
allLTypes = false;
|
2011-10-05 04:05:34 +04:00
|
|
|
else if (!lbaseInfo.getNoReturn() && rbaseInfo.getNoReturn())
|
2010-04-28 20:20:58 +04:00
|
|
|
allRTypes = false;
|
2011-10-05 04:05:34 +04:00
|
|
|
// FIXME: some uses, e.g. conditional exprs, really want this to be 'both'.
|
|
|
|
bool NoReturn = lbaseInfo.getNoReturn() || rbaseInfo.getNoReturn();
|
2009-09-09 19:08:12 +04:00
|
|
|
|
2011-06-16 03:02:42 +04:00
|
|
|
FunctionType::ExtInfo einfo = lbaseInfo.withNoReturn(NoReturn);
|
2010-12-14 11:05:40 +03:00
|
|
|
|
2008-08-22 04:56:42 +04:00
|
|
|
if (lproto && rproto) { // two C99 style function prototypes
|
2009-05-28 02:11:52 +04:00
|
|
|
assert(!lproto->hasExceptionSpec() && !rproto->hasExceptionSpec() &&
|
|
|
|
"C++ shouldn't be here");
|
2008-08-22 04:56:42 +04:00
|
|
|
unsigned lproto_nargs = lproto->getNumArgs();
|
|
|
|
unsigned rproto_nargs = rproto->getNumArgs();
|
|
|
|
|
|
|
|
// Compatible functions must have the same number of arguments
|
|
|
|
if (lproto_nargs != rproto_nargs)
|
|
|
|
return QualType();
|
|
|
|
|
|
|
|
// Variadic and non-variadic functions aren't compatible
|
|
|
|
if (lproto->isVariadic() != rproto->isVariadic())
|
|
|
|
return QualType();
|
|
|
|
|
2008-10-26 19:43:14 +03:00
|
|
|
if (lproto->getTypeQuals() != rproto->getTypeQuals())
|
|
|
|
return QualType();
|
|
|
|
|
2011-09-29 01:52:05 +04:00
|
|
|
if (LangOpts.ObjCAutoRefCount &&
|
|
|
|
!FunctionTypesMatchOnNSConsumedAttrs(rproto, lproto))
|
|
|
|
return QualType();
|
|
|
|
|
2008-08-22 04:56:42 +04:00
|
|
|
// Check argument compatibility
|
2011-07-23 14:55:15 +04:00
|
|
|
SmallVector<QualType, 10> types;
|
2008-08-22 04:56:42 +04:00
|
|
|
for (unsigned i = 0; i < lproto_nargs; i++) {
|
|
|
|
QualType largtype = lproto->getArgType(i).getUnqualifiedType();
|
|
|
|
QualType rargtype = rproto->getArgType(i).getUnqualifiedType();
|
2010-10-24 22:30:18 +04:00
|
|
|
QualType argtype = mergeFunctionArgumentTypes(largtype, rargtype,
|
|
|
|
OfBlockPointer,
|
|
|
|
Unqualified);
|
2008-08-22 04:56:42 +04:00
|
|
|
if (argtype.isNull()) return QualType();
|
2010-07-29 19:18:02 +04:00
|
|
|
|
|
|
|
if (Unqualified)
|
|
|
|
argtype = argtype.getUnqualifiedType();
|
|
|
|
|
2008-08-22 04:56:42 +04:00
|
|
|
types.push_back(argtype);
|
2010-07-29 19:18:02 +04:00
|
|
|
if (Unqualified) {
|
|
|
|
largtype = largtype.getUnqualifiedType();
|
|
|
|
rargtype = rargtype.getUnqualifiedType();
|
|
|
|
}
|
|
|
|
|
2008-10-05 21:34:18 +04:00
|
|
|
if (getCanonicalType(argtype) != getCanonicalType(largtype))
|
|
|
|
allLTypes = false;
|
|
|
|
if (getCanonicalType(argtype) != getCanonicalType(rargtype))
|
|
|
|
allRTypes = false;
|
2008-08-22 04:56:42 +04:00
|
|
|
}
|
2011-09-29 01:52:05 +04:00
|
|
|
|
2008-08-22 04:56:42 +04:00
|
|
|
if (allLTypes) return lhs;
|
|
|
|
if (allRTypes) return rhs;
|
2010-12-14 11:05:40 +03:00
|
|
|
|
|
|
|
FunctionProtoType::ExtProtoInfo EPI = lproto->getExtProtoInfo();
|
|
|
|
EPI.ExtInfo = einfo;
|
|
|
|
return getFunctionType(retType, types.begin(), types.size(), EPI);
|
2008-08-22 04:56:42 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
if (lproto) allRTypes = false;
|
|
|
|
if (rproto) allLTypes = false;
|
|
|
|
|
2009-02-27 02:50:07 +03:00
|
|
|
const FunctionProtoType *proto = lproto ? lproto : rproto;
|
2008-08-22 04:56:42 +04:00
|
|
|
if (proto) {
|
2009-05-28 02:11:52 +04:00
|
|
|
assert(!proto->hasExceptionSpec() && "C++ shouldn't be here");
|
2008-08-22 04:56:42 +04:00
|
|
|
if (proto->isVariadic()) return QualType();
|
|
|
|
// Check that the types are compatible with the types that
|
|
|
|
// would result from default argument promotions (C99 6.7.5.3p15).
|
|
|
|
// The only types actually affected are promotable integer
|
|
|
|
// types and floats, which would be passed as a different
|
|
|
|
// type depending on whether the prototype is visible.
|
|
|
|
unsigned proto_nargs = proto->getNumArgs();
|
|
|
|
for (unsigned i = 0; i < proto_nargs; ++i) {
|
|
|
|
QualType argTy = proto->getArgType(i);
|
2010-02-03 22:27:29 +03:00
|
|
|
|
|
|
|
// Look at the promotion type of enum types, since that is the type used
|
|
|
|
// to pass enum values.
|
|
|
|
if (const EnumType *Enum = argTy->getAs<EnumType>())
|
|
|
|
argTy = Enum->getDecl()->getPromotionType();
|
|
|
|
|
2008-08-22 04:56:42 +04:00
|
|
|
if (argTy->isPromotableIntegerType() ||
|
|
|
|
getCanonicalType(argTy).getUnqualifiedType() == FloatTy)
|
|
|
|
return QualType();
|
|
|
|
}
|
|
|
|
|
|
|
|
if (allLTypes) return lhs;
|
|
|
|
if (allRTypes) return rhs;
|
2010-12-14 11:05:40 +03:00
|
|
|
|
|
|
|
FunctionProtoType::ExtProtoInfo EPI = proto->getExtProtoInfo();
|
|
|
|
EPI.ExtInfo = einfo;
|
2008-08-22 04:56:42 +04:00
|
|
|
return getFunctionType(retType, proto->arg_type_begin(),
|
2010-12-14 11:05:40 +03:00
|
|
|
proto->getNumArgs(), EPI);
|
2008-08-22 04:56:42 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
if (allLTypes) return lhs;
|
|
|
|
if (allRTypes) return rhs;
|
2010-12-15 04:06:38 +03:00
|
|
|
return getFunctionNoProtoType(retType, einfo);
|
2008-08-22 04:56:42 +04:00
|
|
|
}
|
|
|
|
|
2010-03-17 03:20:01 +03:00
|
|
|
QualType ASTContext::mergeTypes(QualType LHS, QualType RHS,
|
2010-07-29 19:18:02 +04:00
|
|
|
bool OfBlockPointer,
|
2011-03-14 19:07:00 +03:00
|
|
|
bool Unqualified, bool BlockReturnType) {
|
2007-12-03 10:33:35 +03:00
|
|
|
// C++ [expr]: If an expression initially has the type "reference to T", the
|
|
|
|
// type is adjusted to "T" prior to any further analysis, the expression
|
|
|
|
// designates the object or function denoted by the reference, and the
|
2009-03-17 02:22:08 +03:00
|
|
|
// expression is an lvalue unless the reference is an rvalue reference and
|
|
|
|
// the expression is a function call (possibly inside parentheses).
|
2010-02-04 00:02:30 +03:00
|
|
|
assert(!LHS->getAs<ReferenceType>() && "LHS is a reference type?");
|
|
|
|
assert(!RHS->getAs<ReferenceType>() && "RHS is a reference type?");
|
2010-07-29 19:18:02 +04:00
|
|
|
|
|
|
|
if (Unqualified) {
|
|
|
|
LHS = LHS.getUnqualifiedType();
|
|
|
|
RHS = RHS.getUnqualifiedType();
|
|
|
|
}
|
2010-02-04 00:02:30 +03:00
|
|
|
|
2008-08-22 04:56:42 +04:00
|
|
|
QualType LHSCan = getCanonicalType(LHS),
|
|
|
|
RHSCan = getCanonicalType(RHS);
|
|
|
|
|
2008-04-07 09:37:56 +04:00
|
|
|
// If two types are identical, they are compatible.
|
2008-08-22 04:56:42 +04:00
|
|
|
if (LHSCan == RHSCan)
|
|
|
|
return LHS;
|
|
|
|
|
2009-09-24 23:53:00 +04:00
|
|
|
// If the qualifiers are different, the types aren't compatible... mostly.
|
First part of changes to eliminate problems with cv-qualifiers and
sugared types. The basic problem is that our qualifier accessors
(getQualifiers, getCVRQualifiers, isConstQualified, etc.) only look at
the current QualType and not at any qualifiers that come from sugared
types, meaning that we won't see these qualifiers through, e.g.,
typedefs:
typedef const int CInt;
typedef CInt Self;
Self.isConstQualified() currently returns false!
Various bugs (e.g., PR5383) have cropped up all over the front end due
to such problems. I'm addressing this problem by splitting each
qualifier accessor into two versions:
- the "local" version only returns qualifiers on this particular
QualType instance
- the "normal" version that will eventually combine qualifiers from this
QualType instance with the qualifiers on the canonical type to
produce the full set of qualifiers.
This commit adds the local versions and switches a few callers from
the "normal" version (e.g., isConstQualified) over to the "local"
version (e.g., isLocalConstQualified) when that is the right thing to
do, e.g., because we're printing or serializing the qualifiers. Also,
switch a bunch of
Context.getCanonicalType(T1).getUnqualifiedType() == Context.getCanonicalType(T2).getQualifiedType()
expressions over to
Context.hasSameUnqualifiedType(T1, T2)
git-svn-id: https://llvm.org/svn/llvm-project/cfe/trunk@88969 91177308-0d34-0410-b5e6-96231b3b80d8
2009-11-17 00:35:15 +03:00
|
|
|
Qualifiers LQuals = LHSCan.getLocalQualifiers();
|
|
|
|
Qualifiers RQuals = RHSCan.getLocalQualifiers();
|
2009-09-24 23:53:00 +04:00
|
|
|
if (LQuals != RQuals) {
|
|
|
|
// If any of these qualifiers are different, we have a type
|
|
|
|
// mismatch.
|
|
|
|
if (LQuals.getCVRQualifiers() != RQuals.getCVRQualifiers() ||
|
2011-06-16 03:02:42 +04:00
|
|
|
LQuals.getAddressSpace() != RQuals.getAddressSpace() ||
|
|
|
|
LQuals.getObjCLifetime() != RQuals.getObjCLifetime())
|
2009-09-24 23:53:00 +04:00
|
|
|
return QualType();
|
|
|
|
|
|
|
|
// Exactly one GC qualifier difference is allowed: __strong is
|
|
|
|
// okay if the other type has no GC qualifier but is an Objective
|
|
|
|
// C object pointer (i.e. implicitly strong by default). We fix
|
|
|
|
// this by pretending that the unqualified type was actually
|
|
|
|
// qualified __strong.
|
|
|
|
Qualifiers::GC GC_L = LQuals.getObjCGCAttr();
|
|
|
|
Qualifiers::GC GC_R = RQuals.getObjCGCAttr();
|
|
|
|
assert((GC_L != GC_R) && "unequal qualifier sets had only equal elements");
|
|
|
|
|
|
|
|
if (GC_L == Qualifiers::Weak || GC_R == Qualifiers::Weak)
|
|
|
|
return QualType();
|
|
|
|
|
|
|
|
if (GC_L == Qualifiers::Strong && RHSCan->isObjCObjectPointerType()) {
|
|
|
|
return mergeTypes(LHS, getObjCGCQualType(RHS, Qualifiers::Strong));
|
|
|
|
}
|
|
|
|
if (GC_R == Qualifiers::Strong && LHSCan->isObjCObjectPointerType()) {
|
|
|
|
return mergeTypes(getObjCGCQualType(LHS, Qualifiers::Strong), RHS);
|
|
|
|
}
|
2008-08-22 04:56:42 +04:00
|
|
|
return QualType();
|
2009-09-24 23:53:00 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
// Okay, qualifiers are equal.
|
2008-08-22 04:56:42 +04:00
|
|
|
|
2009-06-01 05:22:52 +04:00
|
|
|
Type::TypeClass LHSClass = LHSCan->getTypeClass();
|
|
|
|
Type::TypeClass RHSClass = RHSCan->getTypeClass();
|
2008-04-07 09:37:56 +04:00
|
|
|
|
2008-01-14 08:45:46 +03:00
|
|
|
// We want to consider the two function types to be the same for these
|
|
|
|
// comparisons, just force one to the other.
|
|
|
|
if (LHSClass == Type::FunctionProto) LHSClass = Type::FunctionNoProto;
|
|
|
|
if (RHSClass == Type::FunctionProto) RHSClass = Type::FunctionNoProto;
|
2008-02-12 11:23:06 +03:00
|
|
|
|
|
|
|
// Same as above for arrays
|
2008-04-07 09:43:21 +04:00
|
|
|
if (LHSClass == Type::VariableArray || LHSClass == Type::IncompleteArray)
|
|
|
|
LHSClass = Type::ConstantArray;
|
|
|
|
if (RHSClass == Type::VariableArray || RHSClass == Type::IncompleteArray)
|
|
|
|
RHSClass = Type::ConstantArray;
|
2009-09-09 19:08:12 +04:00
|
|
|
|
2010-05-15 15:32:37 +04:00
|
|
|
// ObjCInterfaces are just specialized ObjCObjects.
|
|
|
|
if (LHSClass == Type::ObjCInterface) LHSClass = Type::ObjCObject;
|
|
|
|
if (RHSClass == Type::ObjCInterface) RHSClass = Type::ObjCObject;
|
|
|
|
|
2008-04-19 03:10:10 +04:00
|
|
|
// Canonicalize ExtVector -> Vector.
|
|
|
|
if (LHSClass == Type::ExtVector) LHSClass = Type::Vector;
|
|
|
|
if (RHSClass == Type::ExtVector) RHSClass = Type::Vector;
|
2009-09-09 19:08:12 +04:00
|
|
|
|
2008-12-10 23:07:25 +03:00
|
|
|
// If the canonical type classes don't match.
|
|
|
|
if (LHSClass != RHSClass) {
|
2008-01-14 08:45:46 +03:00
|
|
|
// C99 6.7.2.2p4: Each enumerated type shall be compatible with char,
|
2009-09-09 19:08:12 +04:00
|
|
|
// a signed integer type, or an unsigned integer type.
|
2009-12-09 12:09:27 +03:00
|
|
|
// Compatibility is based on the underlying type, not the promotion
|
|
|
|
// type.
|
2009-09-22 03:43:11 +04:00
|
|
|
if (const EnumType* ETy = LHS->getAs<EnumType>()) {
|
2012-02-06 23:06:20 +04:00
|
|
|
QualType TINT = ETy->getDecl()->getIntegerType();
|
|
|
|
if (!TINT.isNull() && hasSameType(TINT, RHSCan.getUnqualifiedType()))
|
2008-08-22 04:56:42 +04:00
|
|
|
return RHS;
|
2008-02-12 11:46:17 +03:00
|
|
|
}
|
2009-09-22 03:43:11 +04:00
|
|
|
if (const EnumType* ETy = RHS->getAs<EnumType>()) {
|
2012-02-06 23:06:20 +04:00
|
|
|
QualType TINT = ETy->getDecl()->getIntegerType();
|
|
|
|
if (!TINT.isNull() && hasSameType(TINT, LHSCan.getUnqualifiedType()))
|
2008-08-22 04:56:42 +04:00
|
|
|
return LHS;
|
2008-02-12 11:46:17 +03:00
|
|
|
}
|
2012-01-26 04:45:38 +04:00
|
|
|
// allow block pointer type to match an 'id' type.
|
2012-01-26 21:08:50 +04:00
|
|
|
if (OfBlockPointer && !BlockReturnType) {
|
|
|
|
if (LHS->isObjCIdType() && RHS->isBlockPointerType())
|
|
|
|
return LHS;
|
|
|
|
if (RHS->isObjCIdType() && LHS->isBlockPointerType())
|
|
|
|
return RHS;
|
|
|
|
}
|
2012-01-26 04:45:38 +04:00
|
|
|
|
2008-08-22 04:56:42 +04:00
|
|
|
return QualType();
|
2007-10-16 00:41:53 +04:00
|
|
|
}
|
2008-08-22 04:56:42 +04:00
|
|
|
|
2008-01-10 01:43:08 +03:00
|
|
|
// The canonical type classes match.
|
2008-01-14 08:45:46 +03:00
|
|
|
switch (LHSClass) {
|
2009-02-27 02:50:07 +03:00
|
|
|
#define TYPE(Class, Base)
|
|
|
|
#define ABSTRACT_TYPE(Class, Base)
|
2010-03-02 02:49:17 +03:00
|
|
|
#define NON_CANONICAL_UNLESS_DEPENDENT_TYPE(Class, Base) case Type::Class:
|
2009-02-27 02:50:07 +03:00
|
|
|
#define NON_CANONICAL_TYPE(Class, Base) case Type::Class:
|
|
|
|
#define DEPENDENT_TYPE(Class, Base) case Type::Class:
|
|
|
|
#include "clang/AST/TypeNodes.def"
|
2011-09-23 09:06:16 +04:00
|
|
|
llvm_unreachable("Non-canonical and dependent types shouldn't get here");
|
2009-02-27 02:50:07 +03:00
|
|
|
|
2009-03-17 02:22:08 +03:00
|
|
|
case Type::LValueReference:
|
|
|
|
case Type::RValueReference:
|
2009-02-27 02:50:07 +03:00
|
|
|
case Type::MemberPointer:
|
2011-09-23 09:06:16 +04:00
|
|
|
llvm_unreachable("C++ should never be in mergeTypes");
|
2009-02-27 02:50:07 +03:00
|
|
|
|
2010-05-15 15:32:37 +04:00
|
|
|
case Type::ObjCInterface:
|
2009-02-27 02:50:07 +03:00
|
|
|
case Type::IncompleteArray:
|
|
|
|
case Type::VariableArray:
|
|
|
|
case Type::FunctionProto:
|
|
|
|
case Type::ExtVector:
|
2011-09-23 09:06:16 +04:00
|
|
|
llvm_unreachable("Types are eliminated above");
|
2009-02-27 02:50:07 +03:00
|
|
|
|
2008-01-14 08:45:46 +03:00
|
|
|
case Type::Pointer:
|
2008-08-22 04:56:42 +04:00
|
|
|
{
|
|
|
|
// Merge two pointer types, while trying to preserve typedef info
|
2009-07-30 01:53:49 +04:00
|
|
|
QualType LHSPointee = LHS->getAs<PointerType>()->getPointeeType();
|
|
|
|
QualType RHSPointee = RHS->getAs<PointerType>()->getPointeeType();
|
2010-07-29 19:18:02 +04:00
|
|
|
if (Unqualified) {
|
|
|
|
LHSPointee = LHSPointee.getUnqualifiedType();
|
|
|
|
RHSPointee = RHSPointee.getUnqualifiedType();
|
|
|
|
}
|
|
|
|
QualType ResultType = mergeTypes(LHSPointee, RHSPointee, false,
|
|
|
|
Unqualified);
|
2008-08-22 04:56:42 +04:00
|
|
|
if (ResultType.isNull()) return QualType();
|
2009-06-02 09:28:56 +04:00
|
|
|
if (getCanonicalType(LHSPointee) == getCanonicalType(ResultType))
|
2008-10-05 21:34:18 +04:00
|
|
|
return LHS;
|
2009-06-02 09:28:56 +04:00
|
|
|
if (getCanonicalType(RHSPointee) == getCanonicalType(ResultType))
|
2008-10-05 21:34:18 +04:00
|
|
|
return RHS;
|
2008-08-22 04:56:42 +04:00
|
|
|
return getPointerType(ResultType);
|
|
|
|
}
|
2008-12-10 20:49:55 +03:00
|
|
|
case Type::BlockPointer:
|
|
|
|
{
|
|
|
|
// Merge two block pointer types, while trying to preserve typedef info
|
2009-07-30 01:53:49 +04:00
|
|
|
QualType LHSPointee = LHS->getAs<BlockPointerType>()->getPointeeType();
|
|
|
|
QualType RHSPointee = RHS->getAs<BlockPointerType>()->getPointeeType();
|
2010-07-29 19:18:02 +04:00
|
|
|
if (Unqualified) {
|
|
|
|
LHSPointee = LHSPointee.getUnqualifiedType();
|
|
|
|
RHSPointee = RHSPointee.getUnqualifiedType();
|
|
|
|
}
|
|
|
|
QualType ResultType = mergeTypes(LHSPointee, RHSPointee, OfBlockPointer,
|
|
|
|
Unqualified);
|
2008-12-10 20:49:55 +03:00
|
|
|
if (ResultType.isNull()) return QualType();
|
|
|
|
if (getCanonicalType(LHSPointee) == getCanonicalType(ResultType))
|
|
|
|
return LHS;
|
|
|
|
if (getCanonicalType(RHSPointee) == getCanonicalType(ResultType))
|
|
|
|
return RHS;
|
|
|
|
return getBlockPointerType(ResultType);
|
|
|
|
}
|
2011-10-07 03:00:33 +04:00
|
|
|
case Type::Atomic:
|
|
|
|
{
|
|
|
|
// Merge two pointer types, while trying to preserve typedef info
|
|
|
|
QualType LHSValue = LHS->getAs<AtomicType>()->getValueType();
|
|
|
|
QualType RHSValue = RHS->getAs<AtomicType>()->getValueType();
|
|
|
|
if (Unqualified) {
|
|
|
|
LHSValue = LHSValue.getUnqualifiedType();
|
|
|
|
RHSValue = RHSValue.getUnqualifiedType();
|
|
|
|
}
|
|
|
|
QualType ResultType = mergeTypes(LHSValue, RHSValue, false,
|
|
|
|
Unqualified);
|
|
|
|
if (ResultType.isNull()) return QualType();
|
|
|
|
if (getCanonicalType(LHSValue) == getCanonicalType(ResultType))
|
|
|
|
return LHS;
|
|
|
|
if (getCanonicalType(RHSValue) == getCanonicalType(ResultType))
|
|
|
|
return RHS;
|
|
|
|
return getAtomicType(ResultType);
|
|
|
|
}
|
2008-01-14 08:45:46 +03:00
|
|
|
case Type::ConstantArray:
|
2008-08-22 04:56:42 +04:00
|
|
|
{
|
|
|
|
const ConstantArrayType* LCAT = getAsConstantArrayType(LHS);
|
|
|
|
const ConstantArrayType* RCAT = getAsConstantArrayType(RHS);
|
|
|
|
if (LCAT && RCAT && RCAT->getSize() != LCAT->getSize())
|
|
|
|
return QualType();
|
|
|
|
|
|
|
|
QualType LHSElem = getAsArrayType(LHS)->getElementType();
|
|
|
|
QualType RHSElem = getAsArrayType(RHS)->getElementType();
|
2010-07-29 19:18:02 +04:00
|
|
|
if (Unqualified) {
|
|
|
|
LHSElem = LHSElem.getUnqualifiedType();
|
|
|
|
RHSElem = RHSElem.getUnqualifiedType();
|
|
|
|
}
|
|
|
|
|
|
|
|
QualType ResultType = mergeTypes(LHSElem, RHSElem, false, Unqualified);
|
2008-08-22 04:56:42 +04:00
|
|
|
if (ResultType.isNull()) return QualType();
|
2008-10-05 21:34:18 +04:00
|
|
|
if (LCAT && getCanonicalType(LHSElem) == getCanonicalType(ResultType))
|
|
|
|
return LHS;
|
|
|
|
if (RCAT && getCanonicalType(RHSElem) == getCanonicalType(ResultType))
|
|
|
|
return RHS;
|
2008-08-22 05:48:21 +04:00
|
|
|
if (LCAT) return getConstantArrayType(ResultType, LCAT->getSize(),
|
|
|
|
ArrayType::ArraySizeModifier(), 0);
|
|
|
|
if (RCAT) return getConstantArrayType(ResultType, RCAT->getSize(),
|
|
|
|
ArrayType::ArraySizeModifier(), 0);
|
2008-08-22 04:56:42 +04:00
|
|
|
const VariableArrayType* LVAT = getAsVariableArrayType(LHS);
|
|
|
|
const VariableArrayType* RVAT = getAsVariableArrayType(RHS);
|
2008-10-05 21:34:18 +04:00
|
|
|
if (LVAT && getCanonicalType(LHSElem) == getCanonicalType(ResultType))
|
|
|
|
return LHS;
|
|
|
|
if (RVAT && getCanonicalType(RHSElem) == getCanonicalType(ResultType))
|
|
|
|
return RHS;
|
2008-08-22 04:56:42 +04:00
|
|
|
if (LVAT) {
|
|
|
|
// FIXME: This isn't correct! But tricky to implement because
|
|
|
|
// the array's size has to be the size of LHS, but the type
|
|
|
|
// has to be different.
|
|
|
|
return LHS;
|
|
|
|
}
|
|
|
|
if (RVAT) {
|
|
|
|
// FIXME: This isn't correct! But tricky to implement because
|
|
|
|
// the array's size has to be the size of RHS, but the type
|
|
|
|
// has to be different.
|
|
|
|
return RHS;
|
|
|
|
}
|
2008-08-22 05:48:21 +04:00
|
|
|
if (getCanonicalType(LHSElem) == getCanonicalType(ResultType)) return LHS;
|
|
|
|
if (getCanonicalType(RHSElem) == getCanonicalType(ResultType)) return RHS;
|
2009-07-06 19:59:29 +04:00
|
|
|
return getIncompleteArrayType(ResultType,
|
|
|
|
ArrayType::ArraySizeModifier(), 0);
|
2008-08-22 04:56:42 +04:00
|
|
|
}
|
2008-01-14 08:45:46 +03:00
|
|
|
case Type::FunctionNoProto:
|
2010-07-29 19:18:02 +04:00
|
|
|
return mergeFunctionTypes(LHS, RHS, OfBlockPointer, Unqualified);
|
2009-02-27 02:50:07 +03:00
|
|
|
case Type::Record:
|
|
|
|
case Type::Enum:
|
2008-08-22 04:56:42 +04:00
|
|
|
return QualType();
|
2008-01-14 08:45:46 +03:00
|
|
|
case Type::Builtin:
|
2008-04-07 09:55:38 +04:00
|
|
|
// Only exactly equal builtin types are compatible, which is tested above.
|
2008-08-22 04:56:42 +04:00
|
|
|
return QualType();
|
2009-01-29 00:22:12 +03:00
|
|
|
case Type::Complex:
|
|
|
|
// Distinct complex types are incompatible.
|
|
|
|
return QualType();
|
2008-04-07 09:55:38 +04:00
|
|
|
case Type::Vector:
|
2009-02-28 02:04:43 +03:00
|
|
|
// FIXME: The merged type should be an ExtVector!
|
2010-03-13 02:14:13 +03:00
|
|
|
if (areCompatVectorTypes(LHSCan->getAs<VectorType>(),
|
|
|
|
RHSCan->getAs<VectorType>()))
|
2008-08-22 04:56:42 +04:00
|
|
|
return LHS;
|
2008-10-05 21:34:18 +04:00
|
|
|
return QualType();
|
2010-05-15 15:32:37 +04:00
|
|
|
case Type::ObjCObject: {
|
|
|
|
// Check if the types are assignment compatible.
|
2009-02-28 02:04:43 +03:00
|
|
|
// FIXME: This should be type compatibility, e.g. whether
|
|
|
|
// "LHS x; RHS x;" at global scope is legal.
|
2010-05-15 15:32:37 +04:00
|
|
|
const ObjCObjectType* LHSIface = LHS->getAs<ObjCObjectType>();
|
|
|
|
const ObjCObjectType* RHSIface = RHS->getAs<ObjCObjectType>();
|
|
|
|
if (canAssignObjCInterfaces(LHSIface, RHSIface))
|
2009-02-21 19:18:07 +03:00
|
|
|
return LHS;
|
|
|
|
|
2008-08-22 04:56:42 +04:00
|
|
|
return QualType();
|
2009-02-21 20:14:49 +03:00
|
|
|
}
|
2009-07-11 03:34:53 +04:00
|
|
|
case Type::ObjCObjectPointer: {
|
2010-03-17 03:20:01 +03:00
|
|
|
if (OfBlockPointer) {
|
|
|
|
if (canAssignObjCInterfacesInBlockPointer(
|
|
|
|
LHS->getAs<ObjCObjectPointerType>(),
|
2011-03-14 19:07:00 +03:00
|
|
|
RHS->getAs<ObjCObjectPointerType>(),
|
|
|
|
BlockReturnType))
|
2012-01-17 10:56:22 +04:00
|
|
|
return LHS;
|
2010-03-17 03:20:01 +03:00
|
|
|
return QualType();
|
|
|
|
}
|
2009-09-22 03:43:11 +04:00
|
|
|
if (canAssignObjCInterfaces(LHS->getAs<ObjCObjectPointerType>(),
|
|
|
|
RHS->getAs<ObjCObjectPointerType>()))
|
2009-07-11 03:34:53 +04:00
|
|
|
return LHS;
|
|
|
|
|
2008-12-11 01:14:21 +03:00
|
|
|
return QualType();
|
2012-01-17 10:56:22 +04:00
|
|
|
}
|
2007-10-16 00:41:53 +04:00
|
|
|
}
|
2009-02-27 02:50:07 +03:00
|
|
|
|
2012-01-17 10:56:22 +04:00
|
|
|
llvm_unreachable("Invalid Type::Class!");
|
2007-10-16 00:41:53 +04:00
|
|
|
}
|
2007-10-31 20:10:13 +03:00
|
|
|
|
2011-09-29 01:52:05 +04:00
|
|
|
bool ASTContext::FunctionTypesMatchOnNSConsumedAttrs(
|
|
|
|
const FunctionProtoType *FromFunctionType,
|
|
|
|
const FunctionProtoType *ToFunctionType) {
|
|
|
|
if (FromFunctionType->hasAnyConsumedArgs() !=
|
|
|
|
ToFunctionType->hasAnyConsumedArgs())
|
|
|
|
return false;
|
|
|
|
FunctionProtoType::ExtProtoInfo FromEPI =
|
|
|
|
FromFunctionType->getExtProtoInfo();
|
|
|
|
FunctionProtoType::ExtProtoInfo ToEPI =
|
|
|
|
ToFunctionType->getExtProtoInfo();
|
|
|
|
if (FromEPI.ConsumedArguments && ToEPI.ConsumedArguments)
|
|
|
|
for (unsigned ArgIdx = 0, NumArgs = FromFunctionType->getNumArgs();
|
|
|
|
ArgIdx != NumArgs; ++ArgIdx) {
|
|
|
|
if (FromEPI.ConsumedArguments[ArgIdx] !=
|
|
|
|
ToEPI.ConsumedArguments[ArgIdx])
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2010-05-20 01:37:30 +04:00
|
|
|
/// mergeObjCGCQualifiers - This routine merges ObjC's GC attribute of 'LHS' and
|
|
|
|
/// 'RHS' attributes and returns the merged version; including for function
|
|
|
|
/// return types.
|
|
|
|
QualType ASTContext::mergeObjCGCQualifiers(QualType LHS, QualType RHS) {
|
|
|
|
QualType LHSCan = getCanonicalType(LHS),
|
|
|
|
RHSCan = getCanonicalType(RHS);
|
|
|
|
// If two types are identical, they are compatible.
|
|
|
|
if (LHSCan == RHSCan)
|
|
|
|
return LHS;
|
|
|
|
if (RHSCan->isFunctionType()) {
|
|
|
|
if (!LHSCan->isFunctionType())
|
|
|
|
return QualType();
|
|
|
|
QualType OldReturnType =
|
|
|
|
cast<FunctionType>(RHSCan.getTypePtr())->getResultType();
|
|
|
|
QualType NewReturnType =
|
|
|
|
cast<FunctionType>(LHSCan.getTypePtr())->getResultType();
|
|
|
|
QualType ResReturnType =
|
|
|
|
mergeObjCGCQualifiers(NewReturnType, OldReturnType);
|
|
|
|
if (ResReturnType.isNull())
|
|
|
|
return QualType();
|
|
|
|
if (ResReturnType == NewReturnType || ResReturnType == OldReturnType) {
|
|
|
|
// id foo(); ... __strong id foo(); or: __strong id foo(); ... id foo();
|
|
|
|
// In either case, use OldReturnType to build the new function type.
|
|
|
|
const FunctionType *F = LHS->getAs<FunctionType>();
|
|
|
|
if (const FunctionProtoType *FPT = cast<FunctionProtoType>(F)) {
|
2010-12-14 11:05:40 +03:00
|
|
|
FunctionProtoType::ExtProtoInfo EPI = FPT->getExtProtoInfo();
|
|
|
|
EPI.ExtInfo = getFunctionExtInfo(LHS);
|
2010-05-20 01:37:30 +04:00
|
|
|
QualType ResultType
|
|
|
|
= getFunctionType(OldReturnType, FPT->arg_type_begin(),
|
2010-12-14 11:05:40 +03:00
|
|
|
FPT->getNumArgs(), EPI);
|
2010-05-20 01:37:30 +04:00
|
|
|
return ResultType;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return QualType();
|
|
|
|
}
|
|
|
|
|
|
|
|
// If the qualifiers are different, the types can still be merged.
|
|
|
|
Qualifiers LQuals = LHSCan.getLocalQualifiers();
|
|
|
|
Qualifiers RQuals = RHSCan.getLocalQualifiers();
|
|
|
|
if (LQuals != RQuals) {
|
|
|
|
// If any of these qualifiers are different, we have a type mismatch.
|
|
|
|
if (LQuals.getCVRQualifiers() != RQuals.getCVRQualifiers() ||
|
|
|
|
LQuals.getAddressSpace() != RQuals.getAddressSpace())
|
|
|
|
return QualType();
|
|
|
|
|
|
|
|
// Exactly one GC qualifier difference is allowed: __strong is
|
|
|
|
// okay if the other type has no GC qualifier but is an Objective
|
|
|
|
// C object pointer (i.e. implicitly strong by default). We fix
|
|
|
|
// this by pretending that the unqualified type was actually
|
|
|
|
// qualified __strong.
|
|
|
|
Qualifiers::GC GC_L = LQuals.getObjCGCAttr();
|
|
|
|
Qualifiers::GC GC_R = RQuals.getObjCGCAttr();
|
|
|
|
assert((GC_L != GC_R) && "unequal qualifier sets had only equal elements");
|
|
|
|
|
|
|
|
if (GC_L == Qualifiers::Weak || GC_R == Qualifiers::Weak)
|
|
|
|
return QualType();
|
|
|
|
|
|
|
|
if (GC_L == Qualifiers::Strong)
|
|
|
|
return LHS;
|
|
|
|
if (GC_R == Qualifiers::Strong)
|
|
|
|
return RHS;
|
|
|
|
return QualType();
|
|
|
|
}
|
|
|
|
|
|
|
|
if (LHSCan->isObjCObjectPointerType() && RHSCan->isObjCObjectPointerType()) {
|
|
|
|
QualType LHSBaseQT = LHS->getAs<ObjCObjectPointerType>()->getPointeeType();
|
|
|
|
QualType RHSBaseQT = RHS->getAs<ObjCObjectPointerType>()->getPointeeType();
|
|
|
|
QualType ResQT = mergeObjCGCQualifiers(LHSBaseQT, RHSBaseQT);
|
|
|
|
if (ResQT == LHSBaseQT)
|
|
|
|
return LHS;
|
|
|
|
if (ResQT == RHSBaseQT)
|
|
|
|
return RHS;
|
|
|
|
}
|
|
|
|
return QualType();
|
|
|
|
}
|
|
|
|
|
2008-06-28 10:23:08 +04:00
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
// Integer Predicates
|
|
|
|
//===----------------------------------------------------------------------===//
|
2009-01-16 10:15:35 +03:00
|
|
|
|
2011-01-12 12:06:06 +03:00
|
|
|
unsigned ASTContext::getIntWidth(QualType T) const {
|
2011-01-19 09:33:43 +03:00
|
|
|
if (const EnumType *ET = dyn_cast<EnumType>(T))
|
2009-12-11 01:29:29 +03:00
|
|
|
T = ET->getDecl()->getIntegerType();
|
2010-10-09 03:50:27 +04:00
|
|
|
if (T->isBooleanType())
|
|
|
|
return 1;
|
2009-02-13 05:31:07 +03:00
|
|
|
// For builtin types, just use the standard type sizing method
|
2008-06-28 10:23:08 +04:00
|
|
|
return (unsigned)getTypeSize(T);
|
|
|
|
}
|
|
|
|
|
|
|
|
QualType ASTContext::getCorrespondingUnsignedType(QualType T) {
|
2010-07-23 19:58:24 +04:00
|
|
|
assert(T->hasSignedIntegerRepresentation() && "Unexpected type");
|
2009-10-18 00:33:28 +04:00
|
|
|
|
|
|
|
// Turn <4 x signed int> -> <4 x unsigned int>
|
|
|
|
if (const VectorType *VTy = T->getAs<VectorType>())
|
|
|
|
return getVectorType(getCorrespondingUnsignedType(VTy->getElementType()),
|
2010-11-11 00:56:12 +03:00
|
|
|
VTy->getNumElements(), VTy->getVectorKind());
|
2009-10-18 00:33:28 +04:00
|
|
|
|
|
|
|
// For enums, we return the unsigned version of the base type.
|
|
|
|
if (const EnumType *ETy = T->getAs<EnumType>())
|
2008-06-28 10:23:08 +04:00
|
|
|
T = ETy->getDecl()->getIntegerType();
|
2009-10-18 00:33:28 +04:00
|
|
|
|
|
|
|
const BuiltinType *BTy = T->getAs<BuiltinType>();
|
|
|
|
assert(BTy && "Unexpected signed integer type");
|
2008-06-28 10:23:08 +04:00
|
|
|
switch (BTy->getKind()) {
|
|
|
|
case BuiltinType::Char_S:
|
|
|
|
case BuiltinType::SChar:
|
|
|
|
return UnsignedCharTy;
|
|
|
|
case BuiltinType::Short:
|
|
|
|
return UnsignedShortTy;
|
|
|
|
case BuiltinType::Int:
|
|
|
|
return UnsignedIntTy;
|
|
|
|
case BuiltinType::Long:
|
|
|
|
return UnsignedLongTy;
|
|
|
|
case BuiltinType::LongLong:
|
|
|
|
return UnsignedLongLongTy;
|
2009-04-30 06:43:43 +04:00
|
|
|
case BuiltinType::Int128:
|
|
|
|
return UnsignedInt128Ty;
|
2008-06-28 10:23:08 +04:00
|
|
|
default:
|
2011-09-23 09:06:16 +04:00
|
|
|
llvm_unreachable("Unexpected signed integer type");
|
2008-06-28 10:23:08 +04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2010-10-24 21:26:36 +04:00
|
|
|
ASTMutationListener::~ASTMutationListener() { }
|
|
|
|
|
2009-06-14 04:45:47 +04:00
|
|
|
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
// Builtin Type Computation
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
|
|
|
|
/// DecodeTypeFromStr - This decodes one type descriptor from Str, advancing the
|
2010-10-02 02:42:38 +04:00
|
|
|
/// pointer over the consumed characters. This returns the resultant type. If
|
|
|
|
/// AllowTypeModifiers is false then modifier like * are not parsed, just basic
|
|
|
|
/// types. This allows "v2i*" to be parsed as a pointer to a v2i instead of
|
|
|
|
/// a vector of "i*".
|
2010-10-02 02:53:11 +04:00
|
|
|
///
|
|
|
|
/// RequiresICE is filled in on return to indicate whether the value is required
|
|
|
|
/// to be an Integer Constant Expression.
|
2011-01-12 12:06:06 +03:00
|
|
|
static QualType DecodeTypeFromStr(const char *&Str, const ASTContext &Context,
|
2009-06-14 04:45:47 +04:00
|
|
|
ASTContext::GetBuiltinTypeError &Error,
|
2010-10-02 02:53:11 +04:00
|
|
|
bool &RequiresICE,
|
2010-10-02 02:42:38 +04:00
|
|
|
bool AllowTypeModifiers) {
|
2009-06-14 04:45:47 +04:00
|
|
|
// Modifiers.
|
|
|
|
int HowLong = 0;
|
|
|
|
bool Signed = false, Unsigned = false;
|
2010-10-02 02:53:11 +04:00
|
|
|
RequiresICE = false;
|
2010-10-01 11:13:18 +04:00
|
|
|
|
2010-10-02 02:42:38 +04:00
|
|
|
// Read the prefixed modifiers first.
|
2009-06-14 04:45:47 +04:00
|
|
|
bool Done = false;
|
|
|
|
while (!Done) {
|
|
|
|
switch (*Str++) {
|
2009-09-09 19:08:12 +04:00
|
|
|
default: Done = true; --Str; break;
|
2010-10-01 11:13:18 +04:00
|
|
|
case 'I':
|
2010-10-02 02:53:11 +04:00
|
|
|
RequiresICE = true;
|
2010-10-01 11:13:18 +04:00
|
|
|
break;
|
2009-06-14 04:45:47 +04:00
|
|
|
case 'S':
|
|
|
|
assert(!Unsigned && "Can't use both 'S' and 'U' modifiers!");
|
|
|
|
assert(!Signed && "Can't use 'S' modifier multiple times!");
|
|
|
|
Signed = true;
|
|
|
|
break;
|
|
|
|
case 'U':
|
|
|
|
assert(!Signed && "Can't use both 'S' and 'U' modifiers!");
|
|
|
|
assert(!Unsigned && "Can't use 'S' modifier multiple times!");
|
|
|
|
Unsigned = true;
|
|
|
|
break;
|
|
|
|
case 'L':
|
|
|
|
assert(HowLong <= 2 && "Can't have LLLL modifier");
|
|
|
|
++HowLong;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
QualType Type;
|
2009-09-09 19:08:12 +04:00
|
|
|
|
2009-06-14 04:45:47 +04:00
|
|
|
// Read the base type.
|
|
|
|
switch (*Str++) {
|
2011-09-23 09:06:16 +04:00
|
|
|
default: llvm_unreachable("Unknown builtin type letter!");
|
2009-06-14 04:45:47 +04:00
|
|
|
case 'v':
|
|
|
|
assert(HowLong == 0 && !Signed && !Unsigned &&
|
|
|
|
"Bad modifiers used with 'v'!");
|
|
|
|
Type = Context.VoidTy;
|
|
|
|
break;
|
|
|
|
case 'f':
|
|
|
|
assert(HowLong == 0 && !Signed && !Unsigned &&
|
|
|
|
"Bad modifiers used with 'f'!");
|
|
|
|
Type = Context.FloatTy;
|
|
|
|
break;
|
|
|
|
case 'd':
|
|
|
|
assert(HowLong < 2 && !Signed && !Unsigned &&
|
|
|
|
"Bad modifiers used with 'd'!");
|
|
|
|
if (HowLong)
|
|
|
|
Type = Context.LongDoubleTy;
|
|
|
|
else
|
|
|
|
Type = Context.DoubleTy;
|
|
|
|
break;
|
|
|
|
case 's':
|
|
|
|
assert(HowLong == 0 && "Bad modifiers used with 's'!");
|
|
|
|
if (Unsigned)
|
|
|
|
Type = Context.UnsignedShortTy;
|
|
|
|
else
|
|
|
|
Type = Context.ShortTy;
|
|
|
|
break;
|
|
|
|
case 'i':
|
|
|
|
if (HowLong == 3)
|
|
|
|
Type = Unsigned ? Context.UnsignedInt128Ty : Context.Int128Ty;
|
|
|
|
else if (HowLong == 2)
|
|
|
|
Type = Unsigned ? Context.UnsignedLongLongTy : Context.LongLongTy;
|
|
|
|
else if (HowLong == 1)
|
|
|
|
Type = Unsigned ? Context.UnsignedLongTy : Context.LongTy;
|
|
|
|
else
|
|
|
|
Type = Unsigned ? Context.UnsignedIntTy : Context.IntTy;
|
|
|
|
break;
|
|
|
|
case 'c':
|
|
|
|
assert(HowLong == 0 && "Bad modifiers used with 'c'!");
|
|
|
|
if (Signed)
|
|
|
|
Type = Context.SignedCharTy;
|
|
|
|
else if (Unsigned)
|
|
|
|
Type = Context.UnsignedCharTy;
|
|
|
|
else
|
|
|
|
Type = Context.CharTy;
|
|
|
|
break;
|
|
|
|
case 'b': // boolean
|
|
|
|
assert(HowLong == 0 && !Signed && !Unsigned && "Bad modifiers for 'b'!");
|
|
|
|
Type = Context.BoolTy;
|
|
|
|
break;
|
|
|
|
case 'z': // size_t.
|
|
|
|
assert(HowLong == 0 && !Signed && !Unsigned && "Bad modifiers for 'z'!");
|
|
|
|
Type = Context.getSizeType();
|
|
|
|
break;
|
|
|
|
case 'F':
|
|
|
|
Type = Context.getCFConstantStringType();
|
|
|
|
break;
|
2010-11-10 00:38:20 +03:00
|
|
|
case 'G':
|
|
|
|
Type = Context.getObjCIdType();
|
|
|
|
break;
|
|
|
|
case 'H':
|
|
|
|
Type = Context.getObjCSelType();
|
|
|
|
break;
|
2009-06-14 04:45:47 +04:00
|
|
|
case 'a':
|
|
|
|
Type = Context.getBuiltinVaListType();
|
|
|
|
assert(!Type.isNull() && "builtin va list type not initialized!");
|
|
|
|
break;
|
|
|
|
case 'A':
|
|
|
|
// This is a "reference" to a va_list; however, what exactly
|
|
|
|
// this means depends on how va_list is defined. There are two
|
|
|
|
// different kinds of va_list: ones passed by value, and ones
|
|
|
|
// passed by reference. An example of a by-value va_list is
|
|
|
|
// x86, where va_list is a char*. An example of by-ref va_list
|
|
|
|
// is x86-64, where va_list is a __va_list_tag[1]. For x86,
|
|
|
|
// we want this argument to be a char*&; for x86-64, we want
|
|
|
|
// it to be a __va_list_tag*.
|
|
|
|
Type = Context.getBuiltinVaListType();
|
|
|
|
assert(!Type.isNull() && "builtin va list type not initialized!");
|
2010-10-02 02:53:11 +04:00
|
|
|
if (Type->isArrayType())
|
2009-06-14 04:45:47 +04:00
|
|
|
Type = Context.getArrayDecayedType(Type);
|
2010-10-02 02:53:11 +04:00
|
|
|
else
|
2009-06-14 04:45:47 +04:00
|
|
|
Type = Context.getLValueReferenceType(Type);
|
|
|
|
break;
|
|
|
|
case 'V': {
|
|
|
|
char *End;
|
|
|
|
unsigned NumElements = strtoul(Str, &End, 10);
|
|
|
|
assert(End != Str && "Missing vector size");
|
|
|
|
Str = End;
|
2009-09-09 19:08:12 +04:00
|
|
|
|
2010-10-02 02:53:11 +04:00
|
|
|
QualType ElementType = DecodeTypeFromStr(Str, Context, Error,
|
|
|
|
RequiresICE, false);
|
|
|
|
assert(!RequiresICE && "Can't require vector ICE");
|
2010-10-02 02:42:38 +04:00
|
|
|
|
|
|
|
// TODO: No way to make AltiVec vectors in builtins yet.
|
2010-06-23 10:00:24 +04:00
|
|
|
Type = Context.getVectorType(ElementType, NumElements,
|
2010-11-11 00:56:12 +03:00
|
|
|
VectorType::GenericVector);
|
2009-06-14 04:45:47 +04:00
|
|
|
break;
|
|
|
|
}
|
2009-09-29 01:45:01 +04:00
|
|
|
case 'X': {
|
2010-10-02 02:53:11 +04:00
|
|
|
QualType ElementType = DecodeTypeFromStr(Str, Context, Error, RequiresICE,
|
|
|
|
false);
|
|
|
|
assert(!RequiresICE && "Can't require complex ICE");
|
2009-09-29 01:45:01 +04:00
|
|
|
Type = Context.getComplexType(ElementType);
|
|
|
|
break;
|
2011-08-24 03:33:09 +04:00
|
|
|
}
|
|
|
|
case 'Y' : {
|
|
|
|
Type = Context.getPointerDiffType();
|
|
|
|
break;
|
|
|
|
}
|
2009-07-29 02:49:34 +04:00
|
|
|
case 'P':
|
2009-07-07 20:35:42 +04:00
|
|
|
Type = Context.getFILEType();
|
|
|
|
if (Type.isNull()) {
|
2009-07-29 03:57:15 +04:00
|
|
|
Error = ASTContext::GE_Missing_stdio;
|
2009-06-14 04:45:47 +04:00
|
|
|
return QualType();
|
|
|
|
}
|
2009-07-29 03:47:15 +04:00
|
|
|
break;
|
2009-07-29 02:49:34 +04:00
|
|
|
case 'J':
|
2009-07-29 03:57:15 +04:00
|
|
|
if (Signed)
|
2009-07-28 06:25:19 +04:00
|
|
|
Type = Context.getsigjmp_bufType();
|
2009-07-29 03:57:15 +04:00
|
|
|
else
|
|
|
|
Type = Context.getjmp_bufType();
|
|
|
|
|
2009-07-29 03:47:15 +04:00
|
|
|
if (Type.isNull()) {
|
2009-07-29 03:57:15 +04:00
|
|
|
Error = ASTContext::GE_Missing_setjmp;
|
2009-07-29 03:47:15 +04:00
|
|
|
return QualType();
|
|
|
|
}
|
|
|
|
break;
|
2011-11-14 01:51:09 +04:00
|
|
|
case 'K':
|
|
|
|
assert(HowLong == 0 && !Signed && !Unsigned && "Bad modifiers for 'K'!");
|
|
|
|
Type = Context.getucontext_tType();
|
|
|
|
|
|
|
|
if (Type.isNull()) {
|
|
|
|
Error = ASTContext::GE_Missing_ucontext;
|
|
|
|
return QualType();
|
|
|
|
}
|
|
|
|
break;
|
2009-07-28 06:25:19 +04:00
|
|
|
}
|
2009-09-09 19:08:12 +04:00
|
|
|
|
2010-10-02 02:42:38 +04:00
|
|
|
// If there are modifiers and if we're allowed to parse them, go for it.
|
|
|
|
Done = !AllowTypeModifiers;
|
2009-06-14 04:45:47 +04:00
|
|
|
while (!Done) {
|
2010-03-12 07:21:28 +03:00
|
|
|
switch (char c = *Str++) {
|
2010-10-02 02:42:38 +04:00
|
|
|
default: Done = true; --Str; break;
|
|
|
|
case '*':
|
|
|
|
case '&': {
|
|
|
|
// Both pointers and references can have their pointee types
|
|
|
|
// qualified with an address space.
|
|
|
|
char *End;
|
|
|
|
unsigned AddrSpace = strtoul(Str, &End, 10);
|
|
|
|
if (End != Str && AddrSpace != 0) {
|
|
|
|
Type = Context.getAddrSpaceQualType(Type, AddrSpace);
|
|
|
|
Str = End;
|
|
|
|
}
|
|
|
|
if (c == '*')
|
|
|
|
Type = Context.getPointerType(Type);
|
|
|
|
else
|
|
|
|
Type = Context.getLValueReferenceType(Type);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
// FIXME: There's no way to have a built-in with an rvalue ref arg.
|
|
|
|
case 'C':
|
|
|
|
Type = Type.withConst();
|
|
|
|
break;
|
|
|
|
case 'D':
|
|
|
|
Type = Context.getVolatileType(Type);
|
|
|
|
break;
|
2012-01-21 01:40:12 +04:00
|
|
|
case 'R':
|
|
|
|
Type = Type.withRestrict();
|
|
|
|
break;
|
2009-06-14 04:45:47 +04:00
|
|
|
}
|
|
|
|
}
|
2010-10-01 11:13:18 +04:00
|
|
|
|
2010-10-02 02:53:11 +04:00
|
|
|
assert((!RequiresICE || Type->isIntegralOrEnumerationType()) &&
|
2010-10-01 11:13:18 +04:00
|
|
|
"Integer constant 'I' type must be an integer");
|
2009-09-09 19:08:12 +04:00
|
|
|
|
2009-06-14 04:45:47 +04:00
|
|
|
return Type;
|
|
|
|
}
|
|
|
|
|
|
|
|
/// GetBuiltinType - Return the type for the specified builtin.
|
2010-10-02 02:42:38 +04:00
|
|
|
QualType ASTContext::GetBuiltinType(unsigned Id,
|
2010-10-02 02:53:11 +04:00
|
|
|
GetBuiltinTypeError &Error,
|
2011-01-12 12:06:06 +03:00
|
|
|
unsigned *IntegerConstantArgs) const {
|
2010-10-02 02:42:38 +04:00
|
|
|
const char *TypeStr = BuiltinInfo.GetTypeString(Id);
|
2009-09-09 19:08:12 +04:00
|
|
|
|
2011-07-23 14:55:15 +04:00
|
|
|
SmallVector<QualType, 8> ArgTypes;
|
2009-09-09 19:08:12 +04:00
|
|
|
|
2010-10-02 02:53:11 +04:00
|
|
|
bool RequiresICE = false;
|
2009-06-14 04:45:47 +04:00
|
|
|
Error = GE_None;
|
2010-10-02 02:53:11 +04:00
|
|
|
QualType ResType = DecodeTypeFromStr(TypeStr, *this, Error,
|
|
|
|
RequiresICE, true);
|
2009-06-14 04:45:47 +04:00
|
|
|
if (Error != GE_None)
|
|
|
|
return QualType();
|
2010-10-02 02:53:11 +04:00
|
|
|
|
|
|
|
assert(!RequiresICE && "Result of intrinsic cannot be required to be an ICE");
|
|
|
|
|
2009-06-14 04:45:47 +04:00
|
|
|
while (TypeStr[0] && TypeStr[0] != '.') {
|
2010-10-02 02:53:11 +04:00
|
|
|
QualType Ty = DecodeTypeFromStr(TypeStr, *this, Error, RequiresICE, true);
|
2009-06-14 04:45:47 +04:00
|
|
|
if (Error != GE_None)
|
|
|
|
return QualType();
|
|
|
|
|
2010-10-02 02:53:11 +04:00
|
|
|
// If this argument is required to be an IntegerConstantExpression and the
|
|
|
|
// caller cares, fill in the bitmask we return.
|
|
|
|
if (RequiresICE && IntegerConstantArgs)
|
|
|
|
*IntegerConstantArgs |= 1 << ArgTypes.size();
|
|
|
|
|
2009-06-14 04:45:47 +04:00
|
|
|
// Do array -> pointer decay. The builtin should use the decayed type.
|
|
|
|
if (Ty->isArrayType())
|
|
|
|
Ty = getArrayDecayedType(Ty);
|
2009-09-09 19:08:12 +04:00
|
|
|
|
2009-06-14 04:45:47 +04:00
|
|
|
ArgTypes.push_back(Ty);
|
|
|
|
}
|
|
|
|
|
|
|
|
assert((TypeStr[0] != '.' || TypeStr[1] == 0) &&
|
|
|
|
"'.' should only occur at end of builtin type list!");
|
|
|
|
|
2010-12-21 03:44:39 +03:00
|
|
|
FunctionType::ExtInfo EI;
|
|
|
|
if (BuiltinInfo.isNoReturn(Id)) EI = EI.withNoReturn(true);
|
|
|
|
|
|
|
|
bool Variadic = (TypeStr[0] == '.');
|
|
|
|
|
|
|
|
// We really shouldn't be making a no-proto type here, especially in C++.
|
|
|
|
if (ArgTypes.empty() && Variadic)
|
|
|
|
return getFunctionNoProtoType(ResType, EI);
|
2010-02-22 01:15:06 +03:00
|
|
|
|
2010-12-14 11:05:40 +03:00
|
|
|
FunctionProtoType::ExtProtoInfo EPI;
|
2010-12-21 03:44:39 +03:00
|
|
|
EPI.ExtInfo = EI;
|
|
|
|
EPI.Variadic = Variadic;
|
2010-12-14 11:05:40 +03:00
|
|
|
|
|
|
|
return getFunctionType(ResType, ArgTypes.data(), ArgTypes.size(), EPI);
|
2009-06-14 04:45:47 +04:00
|
|
|
}
|
2009-08-19 11:44:53 +04:00
|
|
|
|
2010-07-29 22:15:58 +04:00
|
|
|
GVALinkage ASTContext::GetGVALinkageForFunction(const FunctionDecl *FD) {
|
|
|
|
GVALinkage External = GVA_StrongExternal;
|
|
|
|
|
|
|
|
Linkage L = FD->getLinkage();
|
|
|
|
switch (L) {
|
|
|
|
case NoLinkage:
|
|
|
|
case InternalLinkage:
|
|
|
|
case UniqueExternalLinkage:
|
|
|
|
return GVA_Internal;
|
|
|
|
|
|
|
|
case ExternalLinkage:
|
|
|
|
switch (FD->getTemplateSpecializationKind()) {
|
|
|
|
case TSK_Undeclared:
|
|
|
|
case TSK_ExplicitSpecialization:
|
|
|
|
External = GVA_StrongExternal;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case TSK_ExplicitInstantiationDefinition:
|
|
|
|
return GVA_ExplicitTemplateInstantiation;
|
|
|
|
|
|
|
|
case TSK_ExplicitInstantiationDeclaration:
|
|
|
|
case TSK_ImplicitInstantiation:
|
|
|
|
External = GVA_TemplateInstantiation;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!FD->isInlined())
|
|
|
|
return External;
|
|
|
|
|
2012-03-11 11:00:24 +04:00
|
|
|
if (!getLangOpts().CPlusPlus || FD->hasAttr<GNUInlineAttr>()) {
|
2010-07-29 22:15:58 +04:00
|
|
|
// GNU or C99 inline semantics. Determine whether this symbol should be
|
|
|
|
// externally visible.
|
|
|
|
if (FD->isInlineDefinitionExternallyVisible())
|
|
|
|
return External;
|
|
|
|
|
|
|
|
// C99 inline semantics, where the symbol is not externally visible.
|
|
|
|
return GVA_C99Inline;
|
|
|
|
}
|
|
|
|
|
|
|
|
// C++0x [temp.explicit]p9:
|
|
|
|
// [ Note: The intent is that an inline function that is the subject of
|
|
|
|
// an explicit instantiation declaration will still be implicitly
|
|
|
|
// instantiated when used so that the body can be considered for
|
|
|
|
// inlining, but that no out-of-line copy of the inline function would be
|
|
|
|
// generated in the translation unit. -- end note ]
|
|
|
|
if (FD->getTemplateSpecializationKind()
|
|
|
|
== TSK_ExplicitInstantiationDeclaration)
|
|
|
|
return GVA_C99Inline;
|
|
|
|
|
|
|
|
return GVA_CXXInline;
|
|
|
|
}
|
|
|
|
|
|
|
|
GVALinkage ASTContext::GetGVALinkageForVariable(const VarDecl *VD) {
|
|
|
|
// If this is a static data member, compute the kind of template
|
|
|
|
// specialization. Otherwise, this variable is not part of a
|
|
|
|
// template.
|
|
|
|
TemplateSpecializationKind TSK = TSK_Undeclared;
|
|
|
|
if (VD->isStaticDataMember())
|
|
|
|
TSK = VD->getTemplateSpecializationKind();
|
|
|
|
|
|
|
|
Linkage L = VD->getLinkage();
|
2012-03-11 11:00:24 +04:00
|
|
|
if (L == ExternalLinkage && getLangOpts().CPlusPlus &&
|
2010-07-29 22:15:58 +04:00
|
|
|
VD->getType()->getLinkage() == UniqueExternalLinkage)
|
|
|
|
L = UniqueExternalLinkage;
|
|
|
|
|
|
|
|
switch (L) {
|
|
|
|
case NoLinkage:
|
|
|
|
case InternalLinkage:
|
|
|
|
case UniqueExternalLinkage:
|
|
|
|
return GVA_Internal;
|
|
|
|
|
|
|
|
case ExternalLinkage:
|
|
|
|
switch (TSK) {
|
|
|
|
case TSK_Undeclared:
|
|
|
|
case TSK_ExplicitSpecialization:
|
|
|
|
return GVA_StrongExternal;
|
|
|
|
|
|
|
|
case TSK_ExplicitInstantiationDeclaration:
|
|
|
|
llvm_unreachable("Variable should not be instantiated");
|
|
|
|
// Fall through to treat this like any other instantiation.
|
|
|
|
|
|
|
|
case TSK_ExplicitInstantiationDefinition:
|
|
|
|
return GVA_ExplicitTemplateInstantiation;
|
|
|
|
|
|
|
|
case TSK_ImplicitInstantiation:
|
|
|
|
return GVA_TemplateInstantiation;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-01-17 10:56:22 +04:00
|
|
|
llvm_unreachable("Invalid Linkage!");
|
2010-07-29 22:15:58 +04:00
|
|
|
}
|
|
|
|
|
2010-07-30 00:08:05 +04:00
|
|
|
bool ASTContext::DeclMustBeEmitted(const Decl *D) {
|
2010-07-29 22:15:58 +04:00
|
|
|
if (const VarDecl *VD = dyn_cast<VarDecl>(D)) {
|
|
|
|
if (!VD->isFileVarDecl())
|
|
|
|
return false;
|
2011-09-14 01:35:00 +04:00
|
|
|
} else if (!isa<FunctionDecl>(D))
|
2010-07-29 22:15:58 +04:00
|
|
|
return false;
|
|
|
|
|
2010-07-30 00:07:52 +04:00
|
|
|
// Weak references don't produce any output by themselves.
|
|
|
|
if (D->hasAttr<WeakRefAttr>())
|
|
|
|
return false;
|
|
|
|
|
2010-07-29 22:15:58 +04:00
|
|
|
// Aliases and used decls are required.
|
|
|
|
if (D->hasAttr<AliasAttr>() || D->hasAttr<UsedAttr>())
|
|
|
|
return true;
|
|
|
|
|
|
|
|
if (const FunctionDecl *FD = dyn_cast<FunctionDecl>(D)) {
|
|
|
|
// Forward declarations aren't required.
|
2011-05-07 00:44:56 +04:00
|
|
|
if (!FD->doesThisDeclarationHaveABody())
|
2011-07-18 09:26:13 +04:00
|
|
|
return FD->doesDeclarationForceExternallyVisibleDefinition();
|
2010-07-29 22:15:58 +04:00
|
|
|
|
|
|
|
// Constructors and destructors are required.
|
|
|
|
if (FD->hasAttr<ConstructorAttr>() || FD->hasAttr<DestructorAttr>())
|
|
|
|
return true;
|
|
|
|
|
|
|
|
// The key function for a class is required.
|
|
|
|
if (const CXXMethodDecl *MD = dyn_cast<CXXMethodDecl>(FD)) {
|
|
|
|
const CXXRecordDecl *RD = MD->getParent();
|
|
|
|
if (MD->isOutOfLine() && RD->isDynamicClass()) {
|
|
|
|
const CXXMethodDecl *KeyFunc = getKeyFunction(RD);
|
|
|
|
if (KeyFunc && KeyFunc->getCanonicalDecl() == MD->getCanonicalDecl())
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
GVALinkage Linkage = GetGVALinkageForFunction(FD);
|
|
|
|
|
|
|
|
// static, static inline, always_inline, and extern inline functions can
|
|
|
|
// always be deferred. Normal inline functions can be deferred in C99/C++.
|
|
|
|
// Implicit template instantiations can also be deferred in C++.
|
|
|
|
if (Linkage == GVA_Internal || Linkage == GVA_C99Inline ||
|
2012-02-02 10:06:34 +04:00
|
|
|
Linkage == GVA_CXXInline || Linkage == GVA_TemplateInstantiation)
|
2010-07-29 22:15:58 +04:00
|
|
|
return false;
|
|
|
|
return true;
|
|
|
|
}
|
2011-09-10 04:22:34 +04:00
|
|
|
|
2010-07-29 22:15:58 +04:00
|
|
|
const VarDecl *VD = cast<VarDecl>(D);
|
|
|
|
assert(VD->isFileVarDecl() && "Expected file scoped var");
|
|
|
|
|
2010-07-30 00:07:52 +04:00
|
|
|
if (VD->isThisDeclarationADefinition() == VarDecl::DeclarationOnly)
|
|
|
|
return false;
|
|
|
|
|
2010-07-29 22:15:58 +04:00
|
|
|
// Structs that have non-trivial constructors or destructors are required.
|
|
|
|
|
|
|
|
// FIXME: Handle references.
|
2011-05-09 22:22:59 +04:00
|
|
|
// FIXME: Be more selective about which constructors we care about.
|
2010-07-29 22:15:58 +04:00
|
|
|
if (const RecordType *RT = VD->getType()->getAs<RecordType>()) {
|
|
|
|
if (const CXXRecordDecl *RD = dyn_cast<CXXRecordDecl>(RT->getDecl())) {
|
2011-05-09 22:22:59 +04:00
|
|
|
if (RD->hasDefinition() && !(RD->hasTrivialDefaultConstructor() &&
|
|
|
|
RD->hasTrivialCopyConstructor() &&
|
|
|
|
RD->hasTrivialMoveConstructor() &&
|
|
|
|
RD->hasTrivialDestructor()))
|
2010-07-29 22:15:58 +04:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
GVALinkage L = GetGVALinkageForVariable(VD);
|
|
|
|
if (L == GVA_Internal || L == GVA_TemplateInstantiation) {
|
|
|
|
if (!(VD->getInit() && VD->getInit()->HasSideEffects(*this)))
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
2010-08-16 07:33:14 +04:00
|
|
|
|
2010-11-09 21:04:24 +03:00
|
|
|
CallingConv ASTContext::getDefaultMethodCallConv() {
|
|
|
|
// Pass through to the C++ ABI object
|
|
|
|
return ABI->getDefaultMethodCallConv();
|
|
|
|
}
|
|
|
|
|
2011-01-12 12:06:06 +03:00
|
|
|
bool ASTContext::isNearlyEmpty(const CXXRecordDecl *RD) const {
|
2010-11-25 04:51:53 +03:00
|
|
|
// Pass through to the C++ ABI object
|
|
|
|
return ABI->isNearlyEmpty(RD);
|
|
|
|
}
|
|
|
|
|
2011-01-13 21:57:25 +03:00
|
|
|
MangleContext *ASTContext::createMangleContext() {
|
2011-09-02 04:18:52 +04:00
|
|
|
switch (Target->getCXXABI()) {
|
2011-01-13 21:57:25 +03:00
|
|
|
case CXXABI_ARM:
|
|
|
|
case CXXABI_Itanium:
|
|
|
|
return createItaniumMangleContext(*this, getDiagnostics());
|
|
|
|
case CXXABI_Microsoft:
|
|
|
|
return createMicrosoftMangleContext(*this, getDiagnostics());
|
|
|
|
}
|
2011-09-23 09:06:16 +04:00
|
|
|
llvm_unreachable("Unsupported ABI");
|
2011-01-13 21:57:25 +03:00
|
|
|
}
|
|
|
|
|
2010-08-16 07:33:14 +04:00
|
|
|
CXXABI::~CXXABI() {}
|
2011-04-28 08:53:38 +04:00
|
|
|
|
|
|
|
size_t ASTContext::getSideTableAllocatedMemory() const {
|
2011-07-27 22:41:12 +04:00
|
|
|
return ASTRecordLayouts.getMemorySize()
|
|
|
|
+ llvm::capacity_in_bytes(ObjCLayouts)
|
|
|
|
+ llvm::capacity_in_bytes(KeyFunctions)
|
|
|
|
+ llvm::capacity_in_bytes(ObjCImpls)
|
|
|
|
+ llvm::capacity_in_bytes(BlockVarCopyInits)
|
|
|
|
+ llvm::capacity_in_bytes(DeclAttrs)
|
|
|
|
+ llvm::capacity_in_bytes(InstantiatedFromStaticDataMember)
|
|
|
|
+ llvm::capacity_in_bytes(InstantiatedFromUsingDecl)
|
|
|
|
+ llvm::capacity_in_bytes(InstantiatedFromUsingShadowDecl)
|
|
|
|
+ llvm::capacity_in_bytes(InstantiatedFromUnnamedFieldDecl)
|
|
|
|
+ llvm::capacity_in_bytes(OverriddenMethods)
|
|
|
|
+ llvm::capacity_in_bytes(Types)
|
2011-08-14 07:52:19 +04:00
|
|
|
+ llvm::capacity_in_bytes(VariableArrayTypes)
|
2011-08-14 18:28:49 +04:00
|
|
|
+ llvm::capacity_in_bytes(ClassScopeSpecializationPattern);
|
2011-04-28 08:53:38 +04:00
|
|
|
}
|
2011-10-06 09:00:56 +04:00
|
|
|
|
2012-02-20 23:44:39 +04:00
|
|
|
unsigned ASTContext::getLambdaManglingNumber(CXXMethodDecl *CallOperator) {
|
|
|
|
CXXRecordDecl *Lambda = CallOperator->getParent();
|
|
|
|
return LambdaMangleContexts[Lambda->getDeclContext()]
|
|
|
|
.getManglingNumber(CallOperator);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2011-10-06 09:00:56 +04:00
|
|
|
void ASTContext::setParameterIndex(const ParmVarDecl *D, unsigned int index) {
|
|
|
|
ParamIndices[D] = index;
|
|
|
|
}
|
|
|
|
|
|
|
|
unsigned ASTContext::getParameterIndex(const ParmVarDecl *D) const {
|
|
|
|
ParameterIndexTable::const_iterator I = ParamIndices.find(D);
|
|
|
|
assert(I != ParamIndices.end() &&
|
|
|
|
"ParmIndices lacks entry set by ParmVarDecl");
|
|
|
|
return I->second;
|
|
|
|
}
|