Change struct forward declarations and definitions to use unique RecordDecls, as opposed to creating a single RecordDecl and reusing it.
This change effects both RecordDecls and CXXRecordDecls, but does not effect EnumDecls (yet).
The motivation of this patch is as follows:
- Capture more source information, necessary for refactoring/rewriting clients.
- Pave the way to resolve ownership issues with RecordDecls with the forthcoming
addition of DeclGroups.
Current caveats:
- Until DeclGroups are in place, we will leak RecordDecls not explicitly
referenced by the AST. For example:
typedef struct { ... } x;
The RecordDecl for the struct will be leaked because the TypedefDecl doesn't
refer to it. This will be solved with DeclGroups.
- This patch also (temporarily) breaks CodeGen. More below.
High-level changes:
- As before, TagType still refers to a TagDecl, but it doesn't own it. When
a struct/union/class is first referenced, a RecordType and RecordDecl are
created for it, and the RecordType refers to that RecordDecl. Later, if
a new RecordDecl is created, the pointer to a RecordDecl in RecordType is
updated to point to the RecordDecl that defines the struct/union/class.
- TagDecl and RecordDecl now how a method 'getDefinition()' to return the
TagDecl*/RecordDecl* that refers to the TagDecl* that defines a particular
enum/struct/class/union. This is useful from going from a RecordDecl* that
defines a forward declaration to the RecordDecl* that provides the actual
definition. Note that this also works for EnumDecls, except that in this case
there is no distinction between forward declarations and definitions (yet).
- Clients should no longer assume that 'isDefinition()' returns true from a
RecordDecl if the corresponding struct/union/class has been defined.
isDefinition() only returns true if a particular RecordDecl is the defining
Decl. Use 'getDefinition()' instead to determine if a struct has been defined.
- The main changes to Sema happen in ActOnTag. To make the changes more
incremental, I split off the processing of enums and structs et al into two
code paths. Enums use the original code path (which is in ActOnTag) and
structs use the ActOnTagStruct. Eventually the two code paths will be merged,
but the idea was to preserve the original logic both for comparison and not to
change the logic for both enums and structs all at once.
- There is NO CHAINING of RecordDecls for the same RecordType. All RecordDecls
that correspond to the same type simply have a pointer to that type. If we
need to figure out what are all the RecordDecls for a given type we can build
a backmap.
- The diff in CXXRecordDecl.[cpp,h] is actually very small; it just mimics the
changes to RecordDecl. For some reason 'svn' marks the entire file as changed.
Why is CodeGen broken:
- Codegen assumes that there is an equivalence between RecordDecl* and
RecordType*. This was true before because we only created one RecordDecl* for
a given RecordType*, but it is no longer true. I believe this shouldn't be too
hard to change, but the patch was big enough as it is.
I have tested this patch on both the clang test suite, and by running the static analyzer over Postgresql and a large Apple-internal project (mix of Objective-C and C).
llvm-svn: 55839
2008-09-06 01:16:31 +08:00
|
|
|
//===--- DeclCXX.cpp - C++ Declaration AST Node Implementation ------------===//
|
|
|
|
//
|
|
|
|
// The LLVM Compiler Infrastructure
|
|
|
|
//
|
|
|
|
// This file is distributed under the University of Illinois Open Source
|
|
|
|
// License. See LICENSE.TXT for details.
|
|
|
|
//
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
//
|
|
|
|
// This file implements the C++ related Decl classes.
|
|
|
|
//
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
|
|
|
|
#include "clang/AST/DeclCXX.h"
|
2009-03-26 05:17:03 +08:00
|
|
|
#include "clang/AST/DeclTemplate.h"
|
Change struct forward declarations and definitions to use unique RecordDecls, as opposed to creating a single RecordDecl and reusing it.
This change effects both RecordDecls and CXXRecordDecls, but does not effect EnumDecls (yet).
The motivation of this patch is as follows:
- Capture more source information, necessary for refactoring/rewriting clients.
- Pave the way to resolve ownership issues with RecordDecls with the forthcoming
addition of DeclGroups.
Current caveats:
- Until DeclGroups are in place, we will leak RecordDecls not explicitly
referenced by the AST. For example:
typedef struct { ... } x;
The RecordDecl for the struct will be leaked because the TypedefDecl doesn't
refer to it. This will be solved with DeclGroups.
- This patch also (temporarily) breaks CodeGen. More below.
High-level changes:
- As before, TagType still refers to a TagDecl, but it doesn't own it. When
a struct/union/class is first referenced, a RecordType and RecordDecl are
created for it, and the RecordType refers to that RecordDecl. Later, if
a new RecordDecl is created, the pointer to a RecordDecl in RecordType is
updated to point to the RecordDecl that defines the struct/union/class.
- TagDecl and RecordDecl now how a method 'getDefinition()' to return the
TagDecl*/RecordDecl* that refers to the TagDecl* that defines a particular
enum/struct/class/union. This is useful from going from a RecordDecl* that
defines a forward declaration to the RecordDecl* that provides the actual
definition. Note that this also works for EnumDecls, except that in this case
there is no distinction between forward declarations and definitions (yet).
- Clients should no longer assume that 'isDefinition()' returns true from a
RecordDecl if the corresponding struct/union/class has been defined.
isDefinition() only returns true if a particular RecordDecl is the defining
Decl. Use 'getDefinition()' instead to determine if a struct has been defined.
- The main changes to Sema happen in ActOnTag. To make the changes more
incremental, I split off the processing of enums and structs et al into two
code paths. Enums use the original code path (which is in ActOnTag) and
structs use the ActOnTagStruct. Eventually the two code paths will be merged,
but the idea was to preserve the original logic both for comparison and not to
change the logic for both enums and structs all at once.
- There is NO CHAINING of RecordDecls for the same RecordType. All RecordDecls
that correspond to the same type simply have a pointer to that type. If we
need to figure out what are all the RecordDecls for a given type we can build
a backmap.
- The diff in CXXRecordDecl.[cpp,h] is actually very small; it just mimics the
changes to RecordDecl. For some reason 'svn' marks the entire file as changed.
Why is CodeGen broken:
- Codegen assumes that there is an equivalence between RecordDecl* and
RecordType*. This was true before because we only created one RecordDecl* for
a given RecordType*, but it is no longer true. I believe this shouldn't be too
hard to change, but the patch was big enough as it is.
I have tested this patch on both the clang test suite, and by running the static analyzer over Postgresql and a large Apple-internal project (mix of Objective-C and C).
llvm-svn: 55839
2008-09-06 01:16:31 +08:00
|
|
|
#include "clang/AST/ASTContext.h"
|
2010-10-25 01:26:36 +08:00
|
|
|
#include "clang/AST/ASTMutationListener.h"
|
2010-09-29 08:15:42 +08:00
|
|
|
#include "clang/AST/CXXInheritance.h"
|
2009-03-14 08:25:26 +08:00
|
|
|
#include "clang/AST/Expr.h"
|
2011-11-01 09:16:03 +08:00
|
|
|
#include "clang/AST/ExprCXX.h"
|
2009-12-03 06:36:29 +08:00
|
|
|
#include "clang/AST/TypeLoc.h"
|
2008-11-13 07:21:09 +08:00
|
|
|
#include "clang/Basic/IdentifierTable.h"
|
2008-12-24 05:31:30 +08:00
|
|
|
#include "llvm/ADT/STLExtras.h"
|
2009-09-13 03:52:10 +08:00
|
|
|
#include "llvm/ADT/SmallPtrSet.h"
|
Change struct forward declarations and definitions to use unique RecordDecls, as opposed to creating a single RecordDecl and reusing it.
This change effects both RecordDecls and CXXRecordDecls, but does not effect EnumDecls (yet).
The motivation of this patch is as follows:
- Capture more source information, necessary for refactoring/rewriting clients.
- Pave the way to resolve ownership issues with RecordDecls with the forthcoming
addition of DeclGroups.
Current caveats:
- Until DeclGroups are in place, we will leak RecordDecls not explicitly
referenced by the AST. For example:
typedef struct { ... } x;
The RecordDecl for the struct will be leaked because the TypedefDecl doesn't
refer to it. This will be solved with DeclGroups.
- This patch also (temporarily) breaks CodeGen. More below.
High-level changes:
- As before, TagType still refers to a TagDecl, but it doesn't own it. When
a struct/union/class is first referenced, a RecordType and RecordDecl are
created for it, and the RecordType refers to that RecordDecl. Later, if
a new RecordDecl is created, the pointer to a RecordDecl in RecordType is
updated to point to the RecordDecl that defines the struct/union/class.
- TagDecl and RecordDecl now how a method 'getDefinition()' to return the
TagDecl*/RecordDecl* that refers to the TagDecl* that defines a particular
enum/struct/class/union. This is useful from going from a RecordDecl* that
defines a forward declaration to the RecordDecl* that provides the actual
definition. Note that this also works for EnumDecls, except that in this case
there is no distinction between forward declarations and definitions (yet).
- Clients should no longer assume that 'isDefinition()' returns true from a
RecordDecl if the corresponding struct/union/class has been defined.
isDefinition() only returns true if a particular RecordDecl is the defining
Decl. Use 'getDefinition()' instead to determine if a struct has been defined.
- The main changes to Sema happen in ActOnTag. To make the changes more
incremental, I split off the processing of enums and structs et al into two
code paths. Enums use the original code path (which is in ActOnTag) and
structs use the ActOnTagStruct. Eventually the two code paths will be merged,
but the idea was to preserve the original logic both for comparison and not to
change the logic for both enums and structs all at once.
- There is NO CHAINING of RecordDecls for the same RecordType. All RecordDecls
that correspond to the same type simply have a pointer to that type. If we
need to figure out what are all the RecordDecls for a given type we can build
a backmap.
- The diff in CXXRecordDecl.[cpp,h] is actually very small; it just mimics the
changes to RecordDecl. For some reason 'svn' marks the entire file as changed.
Why is CodeGen broken:
- Codegen assumes that there is an equivalence between RecordDecl* and
RecordType*. This was true before because we only created one RecordDecl* for
a given RecordType*, but it is no longer true. I believe this shouldn't be too
hard to change, but the patch was big enough as it is.
I have tested this patch on both the clang test suite, and by running the static analyzer over Postgresql and a large Apple-internal project (mix of Objective-C and C).
llvm-svn: 55839
2008-09-06 01:16:31 +08:00
|
|
|
using namespace clang;
|
|
|
|
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
// Decl Allocation/Deallocation Method Implementations
|
|
|
|
//===----------------------------------------------------------------------===//
|
2008-12-06 02:15:24 +08:00
|
|
|
|
2011-12-20 10:48:34 +08:00
|
|
|
void AccessSpecDecl::anchor() { }
|
|
|
|
|
2012-01-06 05:55:30 +08:00
|
|
|
AccessSpecDecl *AccessSpecDecl::CreateDeserialized(ASTContext &C, unsigned ID) {
|
|
|
|
void *Mem = AllocateDeserializedDecl(C, ID, sizeof(AccessSpecDecl));
|
|
|
|
return new (Mem) AccessSpecDecl(EmptyShell());
|
|
|
|
}
|
|
|
|
|
2010-02-05 06:26:26 +08:00
|
|
|
CXXRecordDecl::DefinitionData::DefinitionData(CXXRecordDecl *D)
|
|
|
|
: UserDeclaredConstructor(false), UserDeclaredCopyConstructor(false),
|
2011-05-26 04:50:04 +08:00
|
|
|
UserDeclaredMoveConstructor(false), UserDeclaredCopyAssignment(false),
|
|
|
|
UserDeclaredMoveAssignment(false), UserDeclaredDestructor(false),
|
2009-08-16 06:23:00 +08:00
|
|
|
Aggregate(true), PlainOldData(true), Empty(true), Polymorphic(false),
|
2011-04-30 18:07:30 +08:00
|
|
|
Abstract(false), IsStandardLayout(true), HasNoNonEmptyBases(true),
|
Completely re-implement the core logic behind the __is_standard_layout
type trait. The previous implementation suffered from several problems:
1) It implemented all of the logic in RecordType by walking over every
base and field in a CXXRecordDecl and validating the constraints of
the standard. This made for very straightforward code, but is
extremely inefficient. It also is conceptually wrong, the logic tied
to the C++ definition of standard-layout classes should be in
CXXRecordDecl, not RecordType.
2) To address the performance problems with #1, a cache bit was added to
CXXRecordDecl, and at the completion of every C++ class, the
RecordType was queried to determine if it was a standard layout
class, and that state was cached. Two things went very very wrong
with this. First, the caching version of the query *was never
called*. Even within the recursive steps of the walk over all fields
and bases the caching variant was not called, making each query
a full *recursive* walk. Second, despite the cache not being used, it
was computed for every class declared, even when the trait was never
used in the program. This probably significantly regressed compile
time performance for edge-case files.
3) An ASTContext was required merely to query the type trait because
querying it performed the actual computations.
4) The caching bit wasn't managed correctly (uninitialized).
The new implementation follows the system for all the other traits on
C++ classes by encoding all the state needed in the definition data and
building up the trait incrementally as each base and member are added to
the definition of the class.
The idiosyncracies of the specification of standard-layout classes
requires more state than I would like; currently 5 bits. I could
eliminate one of the bits easily at the expense of both clarity and
resilience of the code. I might be able to eliminate one of the other
bits by computing its state in terms of other state bits in the
definition. I've already done that in one place where there was a fairly
simple way to achieve it.
It's possible some of the bits could be moved out of the definition data
and into some other structure which isn't serialized if the serialized
bloat is a problem. That would preclude serialization of a partial class
declaration, but that's likely already precluded.
Comments on any of these issues welcome.
llvm-svn: 130601
2011-04-30 17:17:45 +08:00
|
|
|
HasPrivateFields(false), HasProtectedFields(false), HasPublicFields(false),
|
2012-01-27 02:28:08 +08:00
|
|
|
HasMutableFields(false), HasOnlyCMembers(true),
|
2012-01-24 00:58:45 +08:00
|
|
|
HasTrivialDefaultConstructor(true),
|
2011-12-22 10:22:31 +08:00
|
|
|
HasConstexprNonCopyMoveConstructor(false),
|
|
|
|
DefaultedDefaultConstructorIsConstexpr(true),
|
|
|
|
DefaultedCopyConstructorIsConstexpr(true),
|
|
|
|
DefaultedMoveConstructorIsConstexpr(true),
|
|
|
|
HasConstexprDefaultConstructor(false), HasConstexprCopyConstructor(false),
|
|
|
|
HasConstexprMoveConstructor(false), HasTrivialCopyConstructor(true),
|
2011-05-10 02:22:59 +08:00
|
|
|
HasTrivialMoveConstructor(true), HasTrivialCopyAssignment(true),
|
|
|
|
HasTrivialMoveAssignment(true), HasTrivialDestructor(true),
|
2012-02-25 15:33:38 +08:00
|
|
|
HasIrrelevantDestructor(true),
|
2011-05-10 02:22:59 +08:00
|
|
|
HasNonLiteralTypeFieldsOrBases(false), ComputedVisibleConversions(false),
|
2011-05-12 06:34:38 +08:00
|
|
|
UserProvidedDefaultConstructor(false), DeclaredDefaultConstructor(false),
|
2011-05-26 04:50:04 +08:00
|
|
|
DeclaredCopyConstructor(false), DeclaredMoveConstructor(false),
|
|
|
|
DeclaredCopyAssignment(false), DeclaredMoveAssignment(false),
|
2011-08-31 03:58:05 +08:00
|
|
|
DeclaredDestructor(false), FailedImplicitMoveConstructor(false),
|
2012-01-07 12:59:52 +08:00
|
|
|
FailedImplicitMoveAssignment(false), IsLambda(false), NumBases(0),
|
|
|
|
NumVBases(0), Bases(), VBases(), Definition(D), FirstFriend(0) {
|
2010-02-05 06:26:26 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
CXXRecordDecl::CXXRecordDecl(Kind K, TagKind TK, DeclContext *DC,
|
2011-03-09 22:09:51 +08:00
|
|
|
SourceLocation StartLoc, SourceLocation IdLoc,
|
|
|
|
IdentifierInfo *Id, CXXRecordDecl *PrevDecl)
|
|
|
|
: RecordDecl(K, TK, DC, StartLoc, IdLoc, Id, PrevDecl),
|
2010-02-05 06:26:26 +08:00
|
|
|
DefinitionData(PrevDecl ? PrevDecl->DefinitionData : 0),
|
2009-03-26 05:17:03 +08:00
|
|
|
TemplateOrInstantiation() { }
|
2008-11-13 07:21:09 +08:00
|
|
|
|
2011-01-12 17:06:06 +08:00
|
|
|
CXXRecordDecl *CXXRecordDecl::Create(const ASTContext &C, TagKind TK,
|
2011-03-09 22:09:51 +08:00
|
|
|
DeclContext *DC, SourceLocation StartLoc,
|
|
|
|
SourceLocation IdLoc, IdentifierInfo *Id,
|
2009-05-16 03:11:46 +08:00
|
|
|
CXXRecordDecl* PrevDecl,
|
|
|
|
bool DelayTypeCreation) {
|
2011-03-09 22:09:51 +08:00
|
|
|
CXXRecordDecl* R = new (C) CXXRecordDecl(CXXRecord, TK, DC, StartLoc, IdLoc,
|
|
|
|
Id, PrevDecl);
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-07-30 07:36:44 +08:00
|
|
|
// FIXME: DelayTypeCreation seems like such a hack
|
2009-05-16 03:11:46 +08:00
|
|
|
if (!DelayTypeCreation)
|
2009-09-09 23:08:12 +08:00
|
|
|
C.getTypeDeclType(R, PrevDecl);
|
Change struct forward declarations and definitions to use unique RecordDecls, as opposed to creating a single RecordDecl and reusing it.
This change effects both RecordDecls and CXXRecordDecls, but does not effect EnumDecls (yet).
The motivation of this patch is as follows:
- Capture more source information, necessary for refactoring/rewriting clients.
- Pave the way to resolve ownership issues with RecordDecls with the forthcoming
addition of DeclGroups.
Current caveats:
- Until DeclGroups are in place, we will leak RecordDecls not explicitly
referenced by the AST. For example:
typedef struct { ... } x;
The RecordDecl for the struct will be leaked because the TypedefDecl doesn't
refer to it. This will be solved with DeclGroups.
- This patch also (temporarily) breaks CodeGen. More below.
High-level changes:
- As before, TagType still refers to a TagDecl, but it doesn't own it. When
a struct/union/class is first referenced, a RecordType and RecordDecl are
created for it, and the RecordType refers to that RecordDecl. Later, if
a new RecordDecl is created, the pointer to a RecordDecl in RecordType is
updated to point to the RecordDecl that defines the struct/union/class.
- TagDecl and RecordDecl now how a method 'getDefinition()' to return the
TagDecl*/RecordDecl* that refers to the TagDecl* that defines a particular
enum/struct/class/union. This is useful from going from a RecordDecl* that
defines a forward declaration to the RecordDecl* that provides the actual
definition. Note that this also works for EnumDecls, except that in this case
there is no distinction between forward declarations and definitions (yet).
- Clients should no longer assume that 'isDefinition()' returns true from a
RecordDecl if the corresponding struct/union/class has been defined.
isDefinition() only returns true if a particular RecordDecl is the defining
Decl. Use 'getDefinition()' instead to determine if a struct has been defined.
- The main changes to Sema happen in ActOnTag. To make the changes more
incremental, I split off the processing of enums and structs et al into two
code paths. Enums use the original code path (which is in ActOnTag) and
structs use the ActOnTagStruct. Eventually the two code paths will be merged,
but the idea was to preserve the original logic both for comparison and not to
change the logic for both enums and structs all at once.
- There is NO CHAINING of RecordDecls for the same RecordType. All RecordDecls
that correspond to the same type simply have a pointer to that type. If we
need to figure out what are all the RecordDecls for a given type we can build
a backmap.
- The diff in CXXRecordDecl.[cpp,h] is actually very small; it just mimics the
changes to RecordDecl. For some reason 'svn' marks the entire file as changed.
Why is CodeGen broken:
- Codegen assumes that there is an equivalence between RecordDecl* and
RecordType*. This was true before because we only created one RecordDecl* for
a given RecordType*, but it is no longer true. I believe this shouldn't be too
hard to change, but the patch was big enough as it is.
I have tested this patch on both the clang test suite, and by running the static analyzer over Postgresql and a large Apple-internal project (mix of Objective-C and C).
llvm-svn: 55839
2008-09-06 01:16:31 +08:00
|
|
|
return R;
|
|
|
|
}
|
|
|
|
|
2012-02-13 23:44:47 +08:00
|
|
|
CXXRecordDecl *CXXRecordDecl::CreateLambda(const ASTContext &C, DeclContext *DC,
|
2012-02-22 03:11:17 +08:00
|
|
|
SourceLocation Loc, bool Dependent) {
|
2012-02-13 23:44:47 +08:00
|
|
|
CXXRecordDecl* R = new (C) CXXRecordDecl(CXXRecord, TTK_Class, DC, Loc, Loc,
|
|
|
|
0, 0);
|
|
|
|
R->IsBeingDefined = true;
|
2012-02-22 03:11:17 +08:00
|
|
|
R->DefinitionData = new (C) struct LambdaDefinitionData(R, Dependent);
|
2012-02-13 23:44:47 +08:00
|
|
|
C.getTypeDeclType(R, /*PrevDecl=*/0);
|
|
|
|
return R;
|
|
|
|
}
|
|
|
|
|
2012-01-06 05:55:30 +08:00
|
|
|
CXXRecordDecl *
|
|
|
|
CXXRecordDecl::CreateDeserialized(const ASTContext &C, unsigned ID) {
|
|
|
|
void *Mem = AllocateDeserializedDecl(C, ID, sizeof(CXXRecordDecl));
|
|
|
|
return new (Mem) CXXRecordDecl(CXXRecord, TTK_Struct, 0, SourceLocation(),
|
|
|
|
SourceLocation(), 0, 0);
|
2010-07-02 19:54:55 +08:00
|
|
|
}
|
|
|
|
|
2009-09-09 23:08:12 +08:00
|
|
|
void
|
2010-02-11 09:30:34 +08:00
|
|
|
CXXRecordDecl::setBases(CXXBaseSpecifier const * const *Bases,
|
2008-10-24 02:13:27 +08:00
|
|
|
unsigned NumBases) {
|
2010-02-11 09:30:34 +08:00
|
|
|
ASTContext &C = getASTContext();
|
2008-11-06 00:20:31 +08:00
|
|
|
|
2010-10-30 06:39:52 +08:00
|
|
|
if (!data().Bases.isOffset() && data().NumBases > 0)
|
|
|
|
C.Deallocate(data().getBases());
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2011-10-19 04:08:55 +08:00
|
|
|
if (NumBases) {
|
|
|
|
// C++ [dcl.init.aggr]p1:
|
|
|
|
// An aggregate is [...] a class with [...] no base classes [...].
|
|
|
|
data().Aggregate = false;
|
|
|
|
|
|
|
|
// C++ [class]p4:
|
|
|
|
// A POD-struct is an aggregate class...
|
|
|
|
data().PlainOldData = false;
|
|
|
|
}
|
|
|
|
|
2010-03-29 13:13:12 +08:00
|
|
|
// The set of seen virtual base types.
|
2010-03-30 03:49:09 +08:00
|
|
|
llvm::SmallPtrSet<CanQualType, 8> SeenVBaseTypes;
|
2010-03-29 13:13:12 +08:00
|
|
|
|
|
|
|
// The virtual bases of this class.
|
2011-07-23 18:55:15 +08:00
|
|
|
SmallVector<const CXXBaseSpecifier *, 8> VBases;
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2010-02-05 06:26:26 +08:00
|
|
|
data().Bases = new(C) CXXBaseSpecifier [NumBases];
|
|
|
|
data().NumBases = NumBases;
|
2009-07-11 04:13:23 +08:00
|
|
|
for (unsigned i = 0; i < NumBases; ++i) {
|
2010-10-30 06:39:52 +08:00
|
|
|
data().getBases()[i] = *Bases[i];
|
2009-07-11 04:13:23 +08:00
|
|
|
// Keep track of inherited vbases for this base class.
|
|
|
|
const CXXBaseSpecifier *Base = Bases[i];
|
|
|
|
QualType BaseType = Base->getType();
|
2010-02-27 08:25:28 +08:00
|
|
|
// Skip dependent types; we can't do any checking on them now.
|
2009-07-11 04:13:23 +08:00
|
|
|
if (BaseType->isDependentType())
|
|
|
|
continue;
|
|
|
|
CXXRecordDecl *BaseClassDecl
|
2009-07-30 05:53:49 +08:00
|
|
|
= cast<CXXRecordDecl>(BaseType->getAs<RecordType>()->getDecl());
|
2010-03-29 13:13:12 +08:00
|
|
|
|
2010-09-29 04:38:10 +08:00
|
|
|
// A class with a non-empty base class is not empty.
|
|
|
|
// FIXME: Standard ref?
|
Completely re-implement the core logic behind the __is_standard_layout
type trait. The previous implementation suffered from several problems:
1) It implemented all of the logic in RecordType by walking over every
base and field in a CXXRecordDecl and validating the constraints of
the standard. This made for very straightforward code, but is
extremely inefficient. It also is conceptually wrong, the logic tied
to the C++ definition of standard-layout classes should be in
CXXRecordDecl, not RecordType.
2) To address the performance problems with #1, a cache bit was added to
CXXRecordDecl, and at the completion of every C++ class, the
RecordType was queried to determine if it was a standard layout
class, and that state was cached. Two things went very very wrong
with this. First, the caching version of the query *was never
called*. Even within the recursive steps of the walk over all fields
and bases the caching variant was not called, making each query
a full *recursive* walk. Second, despite the cache not being used, it
was computed for every class declared, even when the trait was never
used in the program. This probably significantly regressed compile
time performance for edge-case files.
3) An ASTContext was required merely to query the type trait because
querying it performed the actual computations.
4) The caching bit wasn't managed correctly (uninitialized).
The new implementation follows the system for all the other traits on
C++ classes by encoding all the state needed in the definition data and
building up the trait incrementally as each base and member are added to
the definition of the class.
The idiosyncracies of the specification of standard-layout classes
requires more state than I would like; currently 5 bits. I could
eliminate one of the bits easily at the expense of both clarity and
resilience of the code. I might be able to eliminate one of the other
bits by computing its state in terms of other state bits in the
definition. I've already done that in one place where there was a fairly
simple way to achieve it.
It's possible some of the bits could be moved out of the definition data
and into some other structure which isn't serialized if the serialized
bloat is a problem. That would preclude serialization of a partial class
declaration, but that's likely already precluded.
Comments on any of these issues welcome.
llvm-svn: 130601
2011-04-30 17:17:45 +08:00
|
|
|
if (!BaseClassDecl->isEmpty()) {
|
|
|
|
if (!data().Empty) {
|
|
|
|
// C++0x [class]p7:
|
|
|
|
// A standard-layout class is a class that:
|
|
|
|
// [...]
|
|
|
|
// -- either has no non-static data members in the most derived
|
|
|
|
// class and at most one base class with non-static data members,
|
|
|
|
// or has no base classes with non-static data members, and
|
|
|
|
// If this is the second non-empty base, then neither of these two
|
|
|
|
// clauses can be true.
|
2011-04-30 18:07:30 +08:00
|
|
|
data().IsStandardLayout = false;
|
Completely re-implement the core logic behind the __is_standard_layout
type trait. The previous implementation suffered from several problems:
1) It implemented all of the logic in RecordType by walking over every
base and field in a CXXRecordDecl and validating the constraints of
the standard. This made for very straightforward code, but is
extremely inefficient. It also is conceptually wrong, the logic tied
to the C++ definition of standard-layout classes should be in
CXXRecordDecl, not RecordType.
2) To address the performance problems with #1, a cache bit was added to
CXXRecordDecl, and at the completion of every C++ class, the
RecordType was queried to determine if it was a standard layout
class, and that state was cached. Two things went very very wrong
with this. First, the caching version of the query *was never
called*. Even within the recursive steps of the walk over all fields
and bases the caching variant was not called, making each query
a full *recursive* walk. Second, despite the cache not being used, it
was computed for every class declared, even when the trait was never
used in the program. This probably significantly regressed compile
time performance for edge-case files.
3) An ASTContext was required merely to query the type trait because
querying it performed the actual computations.
4) The caching bit wasn't managed correctly (uninitialized).
The new implementation follows the system for all the other traits on
C++ classes by encoding all the state needed in the definition data and
building up the trait incrementally as each base and member are added to
the definition of the class.
The idiosyncracies of the specification of standard-layout classes
requires more state than I would like; currently 5 bits. I could
eliminate one of the bits easily at the expense of both clarity and
resilience of the code. I might be able to eliminate one of the other
bits by computing its state in terms of other state bits in the
definition. I've already done that in one place where there was a fairly
simple way to achieve it.
It's possible some of the bits could be moved out of the definition data
and into some other structure which isn't serialized if the serialized
bloat is a problem. That would preclude serialization of a partial class
declaration, but that's likely already precluded.
Comments on any of these issues welcome.
llvm-svn: 130601
2011-04-30 17:17:45 +08:00
|
|
|
}
|
|
|
|
|
2010-09-29 04:38:10 +08:00
|
|
|
data().Empty = false;
|
Completely re-implement the core logic behind the __is_standard_layout
type trait. The previous implementation suffered from several problems:
1) It implemented all of the logic in RecordType by walking over every
base and field in a CXXRecordDecl and validating the constraints of
the standard. This made for very straightforward code, but is
extremely inefficient. It also is conceptually wrong, the logic tied
to the C++ definition of standard-layout classes should be in
CXXRecordDecl, not RecordType.
2) To address the performance problems with #1, a cache bit was added to
CXXRecordDecl, and at the completion of every C++ class, the
RecordType was queried to determine if it was a standard layout
class, and that state was cached. Two things went very very wrong
with this. First, the caching version of the query *was never
called*. Even within the recursive steps of the walk over all fields
and bases the caching variant was not called, making each query
a full *recursive* walk. Second, despite the cache not being used, it
was computed for every class declared, even when the trait was never
used in the program. This probably significantly regressed compile
time performance for edge-case files.
3) An ASTContext was required merely to query the type trait because
querying it performed the actual computations.
4) The caching bit wasn't managed correctly (uninitialized).
The new implementation follows the system for all the other traits on
C++ classes by encoding all the state needed in the definition data and
building up the trait incrementally as each base and member are added to
the definition of the class.
The idiosyncracies of the specification of standard-layout classes
requires more state than I would like; currently 5 bits. I could
eliminate one of the bits easily at the expense of both clarity and
resilience of the code. I might be able to eliminate one of the other
bits by computing its state in terms of other state bits in the
definition. I've already done that in one place where there was a fairly
simple way to achieve it.
It's possible some of the bits could be moved out of the definition data
and into some other structure which isn't serialized if the serialized
bloat is a problem. That would preclude serialization of a partial class
declaration, but that's likely already precluded.
Comments on any of these issues welcome.
llvm-svn: 130601
2011-04-30 17:17:45 +08:00
|
|
|
data().HasNoNonEmptyBases = false;
|
|
|
|
}
|
2010-09-29 04:38:10 +08:00
|
|
|
|
2010-09-29 04:50:54 +08:00
|
|
|
// C++ [class.virtual]p1:
|
|
|
|
// A class that declares or inherits a virtual function is called a
|
|
|
|
// polymorphic class.
|
|
|
|
if (BaseClassDecl->isPolymorphic())
|
|
|
|
data().Polymorphic = true;
|
2011-04-24 10:49:34 +08:00
|
|
|
|
Completely re-implement the core logic behind the __is_standard_layout
type trait. The previous implementation suffered from several problems:
1) It implemented all of the logic in RecordType by walking over every
base and field in a CXXRecordDecl and validating the constraints of
the standard. This made for very straightforward code, but is
extremely inefficient. It also is conceptually wrong, the logic tied
to the C++ definition of standard-layout classes should be in
CXXRecordDecl, not RecordType.
2) To address the performance problems with #1, a cache bit was added to
CXXRecordDecl, and at the completion of every C++ class, the
RecordType was queried to determine if it was a standard layout
class, and that state was cached. Two things went very very wrong
with this. First, the caching version of the query *was never
called*. Even within the recursive steps of the walk over all fields
and bases the caching variant was not called, making each query
a full *recursive* walk. Second, despite the cache not being used, it
was computed for every class declared, even when the trait was never
used in the program. This probably significantly regressed compile
time performance for edge-case files.
3) An ASTContext was required merely to query the type trait because
querying it performed the actual computations.
4) The caching bit wasn't managed correctly (uninitialized).
The new implementation follows the system for all the other traits on
C++ classes by encoding all the state needed in the definition data and
building up the trait incrementally as each base and member are added to
the definition of the class.
The idiosyncracies of the specification of standard-layout classes
requires more state than I would like; currently 5 bits. I could
eliminate one of the bits easily at the expense of both clarity and
resilience of the code. I might be able to eliminate one of the other
bits by computing its state in terms of other state bits in the
definition. I've already done that in one place where there was a fairly
simple way to achieve it.
It's possible some of the bits could be moved out of the definition data
and into some other structure which isn't serialized if the serialized
bloat is a problem. That would preclude serialization of a partial class
declaration, but that's likely already precluded.
Comments on any of these issues welcome.
llvm-svn: 130601
2011-04-30 17:17:45 +08:00
|
|
|
// C++0x [class]p7:
|
|
|
|
// A standard-layout class is a class that: [...]
|
|
|
|
// -- has no non-standard-layout base classes
|
2011-04-30 18:07:30 +08:00
|
|
|
if (!BaseClassDecl->isStandardLayout())
|
|
|
|
data().IsStandardLayout = false;
|
Completely re-implement the core logic behind the __is_standard_layout
type trait. The previous implementation suffered from several problems:
1) It implemented all of the logic in RecordType by walking over every
base and field in a CXXRecordDecl and validating the constraints of
the standard. This made for very straightforward code, but is
extremely inefficient. It also is conceptually wrong, the logic tied
to the C++ definition of standard-layout classes should be in
CXXRecordDecl, not RecordType.
2) To address the performance problems with #1, a cache bit was added to
CXXRecordDecl, and at the completion of every C++ class, the
RecordType was queried to determine if it was a standard layout
class, and that state was cached. Two things went very very wrong
with this. First, the caching version of the query *was never
called*. Even within the recursive steps of the walk over all fields
and bases the caching variant was not called, making each query
a full *recursive* walk. Second, despite the cache not being used, it
was computed for every class declared, even when the trait was never
used in the program. This probably significantly regressed compile
time performance for edge-case files.
3) An ASTContext was required merely to query the type trait because
querying it performed the actual computations.
4) The caching bit wasn't managed correctly (uninitialized).
The new implementation follows the system for all the other traits on
C++ classes by encoding all the state needed in the definition data and
building up the trait incrementally as each base and member are added to
the definition of the class.
The idiosyncracies of the specification of standard-layout classes
requires more state than I would like; currently 5 bits. I could
eliminate one of the bits easily at the expense of both clarity and
resilience of the code. I might be able to eliminate one of the other
bits by computing its state in terms of other state bits in the
definition. I've already done that in one place where there was a fairly
simple way to achieve it.
It's possible some of the bits could be moved out of the definition data
and into some other structure which isn't serialized if the serialized
bloat is a problem. That would preclude serialization of a partial class
declaration, but that's likely already precluded.
Comments on any of these issues welcome.
llvm-svn: 130601
2011-04-30 17:17:45 +08:00
|
|
|
|
2011-04-24 10:49:34 +08:00
|
|
|
// Record if this base is the first non-literal field or base.
|
|
|
|
if (!hasNonLiteralTypeFieldsOrBases() && !BaseType->isLiteralType())
|
|
|
|
data().HasNonLiteralTypeFieldsOrBases = true;
|
2010-09-29 04:50:54 +08:00
|
|
|
|
2010-03-29 13:13:12 +08:00
|
|
|
// Now go through all virtual bases of this base and add them.
|
2009-09-09 23:08:12 +08:00
|
|
|
for (CXXRecordDecl::base_class_iterator VBase =
|
2009-07-11 04:13:23 +08:00
|
|
|
BaseClassDecl->vbases_begin(),
|
|
|
|
E = BaseClassDecl->vbases_end(); VBase != E; ++VBase) {
|
2010-03-29 13:13:12 +08:00
|
|
|
// Add this base if it's not already in the list.
|
2010-03-30 03:49:09 +08:00
|
|
|
if (SeenVBaseTypes.insert(C.getCanonicalType(VBase->getType())))
|
2010-03-29 13:13:12 +08:00
|
|
|
VBases.push_back(VBase);
|
2009-07-11 04:13:23 +08:00
|
|
|
}
|
2010-03-29 13:13:12 +08:00
|
|
|
|
|
|
|
if (Base->isVirtual()) {
|
|
|
|
// Add this base if it's not already in the list.
|
2010-03-30 03:49:09 +08:00
|
|
|
if (SeenVBaseTypes.insert(C.getCanonicalType(BaseType)))
|
2010-03-29 13:13:12 +08:00
|
|
|
VBases.push_back(Base);
|
2010-09-29 04:38:10 +08:00
|
|
|
|
|
|
|
// C++0x [meta.unary.prop] is_empty:
|
|
|
|
// T is a class type, but not a union type, with ... no virtual base
|
|
|
|
// classes
|
|
|
|
data().Empty = false;
|
2010-09-29 04:50:54 +08:00
|
|
|
|
|
|
|
// C++ [class.ctor]p5:
|
2011-05-10 02:22:59 +08:00
|
|
|
// A default constructor is trivial [...] if:
|
|
|
|
// -- its class has [...] no virtual bases
|
|
|
|
data().HasTrivialDefaultConstructor = false;
|
2011-04-24 07:10:33 +08:00
|
|
|
|
|
|
|
// C++0x [class.copy]p13:
|
|
|
|
// A copy/move constructor for class X is trivial if it is neither
|
|
|
|
// user-provided nor deleted and if
|
|
|
|
// -- class X has no virtual functions and no virtual base classes, and
|
2010-09-29 04:50:54 +08:00
|
|
|
data().HasTrivialCopyConstructor = false;
|
2011-04-24 07:10:33 +08:00
|
|
|
data().HasTrivialMoveConstructor = false;
|
|
|
|
|
|
|
|
// C++0x [class.copy]p27:
|
|
|
|
// A copy/move assignment operator for class X is trivial if it is
|
|
|
|
// neither user-provided nor deleted and if
|
|
|
|
// -- class X has no virtual functions and no virtual base classes, and
|
2010-09-29 04:50:54 +08:00
|
|
|
data().HasTrivialCopyAssignment = false;
|
2011-04-24 07:10:33 +08:00
|
|
|
data().HasTrivialMoveAssignment = false;
|
Completely re-implement the core logic behind the __is_standard_layout
type trait. The previous implementation suffered from several problems:
1) It implemented all of the logic in RecordType by walking over every
base and field in a CXXRecordDecl and validating the constraints of
the standard. This made for very straightforward code, but is
extremely inefficient. It also is conceptually wrong, the logic tied
to the C++ definition of standard-layout classes should be in
CXXRecordDecl, not RecordType.
2) To address the performance problems with #1, a cache bit was added to
CXXRecordDecl, and at the completion of every C++ class, the
RecordType was queried to determine if it was a standard layout
class, and that state was cached. Two things went very very wrong
with this. First, the caching version of the query *was never
called*. Even within the recursive steps of the walk over all fields
and bases the caching variant was not called, making each query
a full *recursive* walk. Second, despite the cache not being used, it
was computed for every class declared, even when the trait was never
used in the program. This probably significantly regressed compile
time performance for edge-case files.
3) An ASTContext was required merely to query the type trait because
querying it performed the actual computations.
4) The caching bit wasn't managed correctly (uninitialized).
The new implementation follows the system for all the other traits on
C++ classes by encoding all the state needed in the definition data and
building up the trait incrementally as each base and member are added to
the definition of the class.
The idiosyncracies of the specification of standard-layout classes
requires more state than I would like; currently 5 bits. I could
eliminate one of the bits easily at the expense of both clarity and
resilience of the code. I might be able to eliminate one of the other
bits by computing its state in terms of other state bits in the
definition. I've already done that in one place where there was a fairly
simple way to achieve it.
It's possible some of the bits could be moved out of the definition data
and into some other structure which isn't serialized if the serialized
bloat is a problem. That would preclude serialization of a partial class
declaration, but that's likely already precluded.
Comments on any of these issues welcome.
llvm-svn: 130601
2011-04-30 17:17:45 +08:00
|
|
|
|
|
|
|
// C++0x [class]p7:
|
|
|
|
// A standard-layout class is a class that: [...]
|
|
|
|
// -- has [...] no virtual base classes
|
2011-04-30 18:07:30 +08:00
|
|
|
data().IsStandardLayout = false;
|
2011-12-22 10:22:31 +08:00
|
|
|
|
|
|
|
// C++11 [dcl.constexpr]p4:
|
|
|
|
// In the definition of a constexpr constructor [...]
|
|
|
|
// -- the class shall not have any virtual base classes
|
|
|
|
data().DefaultedDefaultConstructorIsConstexpr = false;
|
|
|
|
data().DefaultedCopyConstructorIsConstexpr = false;
|
|
|
|
data().DefaultedMoveConstructorIsConstexpr = false;
|
2010-09-29 04:50:54 +08:00
|
|
|
} else {
|
|
|
|
// C++ [class.ctor]p5:
|
2011-05-10 02:22:59 +08:00
|
|
|
// A default constructor is trivial [...] if:
|
|
|
|
// -- all the direct base classes of its class have trivial default
|
|
|
|
// constructors.
|
|
|
|
if (!BaseClassDecl->hasTrivialDefaultConstructor())
|
|
|
|
data().HasTrivialDefaultConstructor = false;
|
2010-09-29 04:50:54 +08:00
|
|
|
|
2011-04-24 07:10:33 +08:00
|
|
|
// C++0x [class.copy]p13:
|
|
|
|
// A copy/move constructor for class X is trivial if [...]
|
|
|
|
// [...]
|
|
|
|
// -- the constructor selected to copy/move each direct base class
|
|
|
|
// subobject is trivial, and
|
|
|
|
// FIXME: C++0x: We need to only consider the selected constructor
|
|
|
|
// instead of all of them.
|
2010-09-29 04:50:54 +08:00
|
|
|
if (!BaseClassDecl->hasTrivialCopyConstructor())
|
|
|
|
data().HasTrivialCopyConstructor = false;
|
2011-04-24 07:10:33 +08:00
|
|
|
if (!BaseClassDecl->hasTrivialMoveConstructor())
|
|
|
|
data().HasTrivialMoveConstructor = false;
|
|
|
|
|
|
|
|
// C++0x [class.copy]p27:
|
|
|
|
// A copy/move assignment operator for class X is trivial if [...]
|
|
|
|
// [...]
|
|
|
|
// -- the assignment operator selected to copy/move each direct base
|
|
|
|
// class subobject is trivial, and
|
|
|
|
// FIXME: C++0x: We need to only consider the selected operator instead
|
|
|
|
// of all of them.
|
2010-09-29 04:50:54 +08:00
|
|
|
if (!BaseClassDecl->hasTrivialCopyAssignment())
|
|
|
|
data().HasTrivialCopyAssignment = false;
|
2011-04-24 07:10:33 +08:00
|
|
|
if (!BaseClassDecl->hasTrivialMoveAssignment())
|
|
|
|
data().HasTrivialMoveAssignment = false;
|
2011-12-22 10:22:31 +08:00
|
|
|
|
|
|
|
// C++11 [class.ctor]p6:
|
2012-01-12 02:26:05 +08:00
|
|
|
// If that user-written default constructor would satisfy the
|
2011-12-22 10:22:31 +08:00
|
|
|
// requirements of a constexpr constructor, the implicitly-defined
|
|
|
|
// default constructor is constexpr.
|
|
|
|
if (!BaseClassDecl->hasConstexprDefaultConstructor())
|
|
|
|
data().DefaultedDefaultConstructorIsConstexpr = false;
|
|
|
|
|
|
|
|
// C++11 [class.copy]p13:
|
|
|
|
// If the implicitly-defined constructor would satisfy the requirements
|
|
|
|
// of a constexpr constructor, the implicitly-defined constructor is
|
|
|
|
// constexpr.
|
|
|
|
// C++11 [dcl.constexpr]p4:
|
|
|
|
// -- every constructor involved in initializing [...] base class
|
|
|
|
// sub-objects shall be a constexpr constructor
|
|
|
|
if (!BaseClassDecl->hasConstexprCopyConstructor())
|
|
|
|
data().DefaultedCopyConstructorIsConstexpr = false;
|
|
|
|
if (BaseClassDecl->hasDeclaredMoveConstructor() ||
|
|
|
|
BaseClassDecl->needsImplicitMoveConstructor())
|
|
|
|
// FIXME: If the implicit move constructor generated for the base class
|
|
|
|
// would be ill-formed, the implicit move constructor generated for the
|
|
|
|
// derived class calls the base class' copy constructor.
|
|
|
|
data().DefaultedMoveConstructorIsConstexpr &=
|
2012-01-12 02:26:05 +08:00
|
|
|
BaseClassDecl->hasConstexprMoveConstructor();
|
2011-12-22 10:22:31 +08:00
|
|
|
else if (!BaseClassDecl->hasConstexprCopyConstructor())
|
|
|
|
data().DefaultedMoveConstructorIsConstexpr = false;
|
2009-07-11 04:13:23 +08:00
|
|
|
}
|
2010-09-29 04:50:54 +08:00
|
|
|
|
|
|
|
// C++ [class.ctor]p3:
|
|
|
|
// A destructor is trivial if all the direct base classes of its class
|
|
|
|
// have trivial destructors.
|
|
|
|
if (!BaseClassDecl->hasTrivialDestructor())
|
|
|
|
data().HasTrivialDestructor = false;
|
2012-02-25 15:33:38 +08:00
|
|
|
|
|
|
|
if (!BaseClassDecl->hasIrrelevantDestructor())
|
|
|
|
data().HasIrrelevantDestructor = false;
|
|
|
|
|
2011-06-16 07:02:42 +08:00
|
|
|
// A class has an Objective-C object member if... or any of its bases
|
|
|
|
// has an Objective-C object member.
|
|
|
|
if (BaseClassDecl->hasObjectMember())
|
|
|
|
setHasObjectMember(true);
|
|
|
|
|
2011-05-13 09:05:07 +08:00
|
|
|
// Keep track of the presence of mutable fields.
|
|
|
|
if (BaseClassDecl->hasMutableFields())
|
|
|
|
data().HasMutableFields = true;
|
2009-07-11 04:13:23 +08:00
|
|
|
}
|
2010-03-29 13:13:12 +08:00
|
|
|
|
|
|
|
if (VBases.empty())
|
|
|
|
return;
|
|
|
|
|
|
|
|
// Create base specifier for any direct or indirect virtual bases.
|
|
|
|
data().VBases = new (C) CXXBaseSpecifier[VBases.size()];
|
|
|
|
data().NumVBases = VBases.size();
|
2011-07-13 07:49:11 +08:00
|
|
|
for (int I = 0, E = VBases.size(); I != E; ++I)
|
|
|
|
data().getVBases()[I] = *VBases[I];
|
2008-10-24 02:13:27 +08:00
|
|
|
}
|
|
|
|
|
2010-01-15 01:47:39 +08:00
|
|
|
/// Callback function for CXXRecordDecl::forallBases that acknowledges
|
|
|
|
/// that it saw a base class.
|
|
|
|
static bool SawBase(const CXXRecordDecl *, void *) {
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool CXXRecordDecl::hasAnyDependentBases() const {
|
|
|
|
if (!isDependentContext())
|
|
|
|
return false;
|
|
|
|
|
|
|
|
return !forallBases(SawBase, 0);
|
|
|
|
}
|
|
|
|
|
2011-05-26 04:50:04 +08:00
|
|
|
bool CXXRecordDecl::hasConstCopyConstructor() const {
|
|
|
|
return getCopyConstructor(Qualifiers::Const) != 0;
|
2009-06-23 07:34:40 +08:00
|
|
|
}
|
|
|
|
|
2011-04-23 18:47:28 +08:00
|
|
|
bool CXXRecordDecl::isTriviallyCopyable() const {
|
|
|
|
// C++0x [class]p5:
|
|
|
|
// A trivially copyable class is a class that:
|
|
|
|
// -- has no non-trivial copy constructors,
|
|
|
|
if (!hasTrivialCopyConstructor()) return false;
|
|
|
|
// -- has no non-trivial move constructors,
|
2011-04-24 07:10:33 +08:00
|
|
|
if (!hasTrivialMoveConstructor()) return false;
|
2011-04-23 18:47:28 +08:00
|
|
|
// -- has no non-trivial copy assignment operators,
|
|
|
|
if (!hasTrivialCopyAssignment()) return false;
|
|
|
|
// -- has no non-trivial move assignment operators, and
|
2011-04-24 07:10:33 +08:00
|
|
|
if (!hasTrivialMoveAssignment()) return false;
|
2011-04-23 18:47:28 +08:00
|
|
|
// -- has a trivial destructor.
|
|
|
|
if (!hasTrivialDestructor()) return false;
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2010-07-02 04:59:04 +08:00
|
|
|
/// \brief Perform a simplistic form of overload resolution that only considers
|
|
|
|
/// cv-qualifiers on a single parameter, and return the best overload candidate
|
|
|
|
/// (if there is one).
|
|
|
|
static CXXMethodDecl *
|
|
|
|
GetBestOverloadCandidateSimple(
|
2011-07-23 18:55:15 +08:00
|
|
|
const SmallVectorImpl<std::pair<CXXMethodDecl *, Qualifiers> > &Cands) {
|
2010-07-02 04:59:04 +08:00
|
|
|
if (Cands.empty())
|
|
|
|
return 0;
|
|
|
|
if (Cands.size() == 1)
|
|
|
|
return Cands[0].first;
|
|
|
|
|
|
|
|
unsigned Best = 0, N = Cands.size();
|
|
|
|
for (unsigned I = 1; I != N; ++I)
|
2011-04-28 08:56:09 +08:00
|
|
|
if (Cands[Best].second.compatiblyIncludes(Cands[I].second))
|
2010-07-02 04:59:04 +08:00
|
|
|
Best = I;
|
|
|
|
|
|
|
|
for (unsigned I = 1; I != N; ++I)
|
2011-04-28 08:56:09 +08:00
|
|
|
if (Cands[Best].second.compatiblyIncludes(Cands[I].second))
|
2010-07-02 04:59:04 +08:00
|
|
|
return 0;
|
|
|
|
|
|
|
|
return Cands[Best].first;
|
|
|
|
}
|
|
|
|
|
2011-05-26 04:50:04 +08:00
|
|
|
CXXConstructorDecl *CXXRecordDecl::getCopyConstructor(unsigned TypeQuals) const{
|
|
|
|
ASTContext &Context = getASTContext();
|
2009-01-06 04:52:13 +08:00
|
|
|
QualType ClassType
|
|
|
|
= Context.getTypeDeclType(const_cast<CXXRecordDecl*>(this));
|
2009-09-09 23:08:12 +08:00
|
|
|
DeclarationName ConstructorName
|
2008-12-16 05:24:18 +08:00
|
|
|
= Context.DeclarationNames.getCXXConstructorName(
|
2009-06-23 07:34:40 +08:00
|
|
|
Context.getCanonicalType(ClassType));
|
|
|
|
unsigned FoundTQs;
|
2011-07-23 18:55:15 +08:00
|
|
|
SmallVector<std::pair<CXXMethodDecl *, Qualifiers>, 4> Found;
|
2008-12-24 05:31:30 +08:00
|
|
|
DeclContext::lookup_const_iterator Con, ConEnd;
|
2009-06-30 10:36:12 +08:00
|
|
|
for (llvm::tie(Con, ConEnd) = this->lookup(ConstructorName);
|
2008-12-24 05:31:30 +08:00
|
|
|
Con != ConEnd; ++Con) {
|
2009-09-04 22:46:39 +08:00
|
|
|
// C++ [class.copy]p2:
|
|
|
|
// A non-template constructor for class X is a copy constructor if [...]
|
|
|
|
if (isa<FunctionTemplateDecl>(*Con))
|
|
|
|
continue;
|
|
|
|
|
2010-07-02 04:59:04 +08:00
|
|
|
CXXConstructorDecl *Constructor = cast<CXXConstructorDecl>(*Con);
|
|
|
|
if (Constructor->isCopyConstructor(FoundTQs)) {
|
2009-09-25 03:53:00 +08:00
|
|
|
if (((TypeQuals & Qualifiers::Const) == (FoundTQs & Qualifiers::Const)) ||
|
|
|
|
(!(TypeQuals & Qualifiers::Const) && (FoundTQs & Qualifiers::Const)))
|
2010-07-02 04:59:04 +08:00
|
|
|
Found.push_back(std::make_pair(
|
|
|
|
const_cast<CXXConstructorDecl *>(Constructor),
|
|
|
|
Qualifiers::fromCVRMask(FoundTQs)));
|
2009-06-23 07:34:40 +08:00
|
|
|
}
|
2008-11-04 01:51:48 +08:00
|
|
|
}
|
2010-07-02 04:59:04 +08:00
|
|
|
|
|
|
|
return cast_or_null<CXXConstructorDecl>(
|
|
|
|
GetBestOverloadCandidateSimple(Found));
|
2008-11-04 01:51:48 +08:00
|
|
|
}
|
|
|
|
|
2011-05-26 04:50:04 +08:00
|
|
|
CXXConstructorDecl *CXXRecordDecl::getMoveConstructor() const {
|
|
|
|
for (ctor_iterator I = ctor_begin(), E = ctor_end(); I != E; ++I)
|
|
|
|
if (I->isMoveConstructor())
|
|
|
|
return *I;
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2010-07-02 01:48:08 +08:00
|
|
|
CXXMethodDecl *CXXRecordDecl::getCopyAssignmentOperator(bool ArgIsConst) const {
|
|
|
|
ASTContext &Context = getASTContext();
|
|
|
|
QualType Class = Context.getTypeDeclType(const_cast<CXXRecordDecl *>(this));
|
|
|
|
DeclarationName Name = Context.DeclarationNames.getCXXOperatorName(OO_Equal);
|
|
|
|
|
2011-07-23 18:55:15 +08:00
|
|
|
SmallVector<std::pair<CXXMethodDecl *, Qualifiers>, 4> Found;
|
2010-07-02 01:48:08 +08:00
|
|
|
DeclContext::lookup_const_iterator Op, OpEnd;
|
|
|
|
for (llvm::tie(Op, OpEnd) = this->lookup(Name); Op != OpEnd; ++Op) {
|
|
|
|
// C++ [class.copy]p9:
|
|
|
|
// A user-declared copy assignment operator is a non-static non-template
|
|
|
|
// member function of class X with exactly one parameter of type X, X&,
|
|
|
|
// const X&, volatile X& or const volatile X&.
|
|
|
|
const CXXMethodDecl* Method = dyn_cast<CXXMethodDecl>(*Op);
|
|
|
|
if (!Method || Method->isStatic() || Method->getPrimaryTemplate())
|
|
|
|
continue;
|
|
|
|
|
|
|
|
const FunctionProtoType *FnType
|
|
|
|
= Method->getType()->getAs<FunctionProtoType>();
|
|
|
|
assert(FnType && "Overloaded operator has no prototype.");
|
|
|
|
// Don't assert on this; an invalid decl might have been left in the AST.
|
|
|
|
if (FnType->getNumArgs() != 1 || FnType->isVariadic())
|
|
|
|
continue;
|
|
|
|
|
|
|
|
QualType ArgType = FnType->getArgType(0);
|
|
|
|
Qualifiers Quals;
|
|
|
|
if (const LValueReferenceType *Ref = ArgType->getAs<LValueReferenceType>()) {
|
|
|
|
ArgType = Ref->getPointeeType();
|
|
|
|
// If we have a const argument and we have a reference to a non-const,
|
|
|
|
// this function does not match.
|
|
|
|
if (ArgIsConst && !ArgType.isConstQualified())
|
|
|
|
continue;
|
|
|
|
|
|
|
|
Quals = ArgType.getQualifiers();
|
|
|
|
} else {
|
|
|
|
// By-value copy-assignment operators are treated like const X&
|
|
|
|
// copy-assignment operators.
|
|
|
|
Quals = Qualifiers::fromCVRMask(Qualifiers::Const);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!Context.hasSameUnqualifiedType(ArgType, Class))
|
|
|
|
continue;
|
|
|
|
|
|
|
|
// Save this copy-assignment operator. It might be "the one".
|
|
|
|
Found.push_back(std::make_pair(const_cast<CXXMethodDecl *>(Method), Quals));
|
|
|
|
}
|
|
|
|
|
|
|
|
// Use a simplistic form of overload resolution to find the candidate.
|
|
|
|
return GetBestOverloadCandidateSimple(Found);
|
|
|
|
}
|
|
|
|
|
2011-05-26 04:50:04 +08:00
|
|
|
CXXMethodDecl *CXXRecordDecl::getMoveAssignmentOperator() const {
|
|
|
|
for (method_iterator I = method_begin(), E = method_end(); I != E; ++I)
|
|
|
|
if (I->isMoveAssignmentOperator())
|
|
|
|
return *I;
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2010-09-29 05:55:22 +08:00
|
|
|
void CXXRecordDecl::markedVirtualFunctionPure() {
|
|
|
|
// C++ [class.abstract]p2:
|
|
|
|
// A class is abstract if it has at least one pure virtual function.
|
|
|
|
data().Abstract = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
void CXXRecordDecl::addedMember(Decl *D) {
|
2012-01-27 02:28:08 +08:00
|
|
|
if (!D->isImplicit() &&
|
|
|
|
!isa<FieldDecl>(D) &&
|
|
|
|
!isa<IndirectFieldDecl>(D) &&
|
|
|
|
(!isa<TagDecl>(D) || cast<TagDecl>(D)->getTagKind() == TTK_Class))
|
|
|
|
data().HasOnlyCMembers = false;
|
2012-01-24 00:58:45 +08:00
|
|
|
|
2010-09-28 06:06:20 +08:00
|
|
|
// Ignore friends and invalid declarations.
|
|
|
|
if (D->getFriendObjectKind() || D->isInvalidDecl())
|
2010-09-28 05:17:54 +08:00
|
|
|
return;
|
|
|
|
|
2010-09-28 06:06:20 +08:00
|
|
|
FunctionTemplateDecl *FunTmpl = dyn_cast<FunctionTemplateDecl>(D);
|
|
|
|
if (FunTmpl)
|
|
|
|
D = FunTmpl->getTemplatedDecl();
|
|
|
|
|
2010-09-29 03:45:33 +08:00
|
|
|
if (CXXMethodDecl *Method = dyn_cast<CXXMethodDecl>(D)) {
|
|
|
|
if (Method->isVirtual()) {
|
|
|
|
// C++ [dcl.init.aggr]p1:
|
|
|
|
// An aggregate is an array or a class with [...] no virtual functions.
|
|
|
|
data().Aggregate = false;
|
|
|
|
|
|
|
|
// C++ [class]p4:
|
|
|
|
// A POD-struct is an aggregate class...
|
|
|
|
data().PlainOldData = false;
|
2010-09-29 04:38:10 +08:00
|
|
|
|
|
|
|
// Virtual functions make the class non-empty.
|
|
|
|
// FIXME: Standard ref?
|
|
|
|
data().Empty = false;
|
2010-09-29 04:50:54 +08:00
|
|
|
|
|
|
|
// C++ [class.virtual]p1:
|
|
|
|
// A class that declares or inherits a virtual function is called a
|
|
|
|
// polymorphic class.
|
|
|
|
data().Polymorphic = true;
|
|
|
|
|
2011-05-10 02:22:59 +08:00
|
|
|
// C++0x [class.ctor]p5
|
|
|
|
// A default constructor is trivial [...] if:
|
|
|
|
// -- its class has no virtual functions [...]
|
|
|
|
data().HasTrivialDefaultConstructor = false;
|
2011-04-24 07:10:33 +08:00
|
|
|
|
|
|
|
// C++0x [class.copy]p13:
|
|
|
|
// A copy/move constructor for class X is trivial if [...]
|
|
|
|
// -- class X has no virtual functions [...]
|
2010-09-29 04:50:54 +08:00
|
|
|
data().HasTrivialCopyConstructor = false;
|
2011-04-24 07:10:33 +08:00
|
|
|
data().HasTrivialMoveConstructor = false;
|
|
|
|
|
|
|
|
// C++0x [class.copy]p27:
|
|
|
|
// A copy/move assignment operator for class X is trivial if [...]
|
|
|
|
// -- class X has no virtual functions [...]
|
2010-09-29 04:50:54 +08:00
|
|
|
data().HasTrivialCopyAssignment = false;
|
2011-04-24 07:10:33 +08:00
|
|
|
data().HasTrivialMoveAssignment = false;
|
2011-11-08 04:56:01 +08:00
|
|
|
|
Completely re-implement the core logic behind the __is_standard_layout
type trait. The previous implementation suffered from several problems:
1) It implemented all of the logic in RecordType by walking over every
base and field in a CXXRecordDecl and validating the constraints of
the standard. This made for very straightforward code, but is
extremely inefficient. It also is conceptually wrong, the logic tied
to the C++ definition of standard-layout classes should be in
CXXRecordDecl, not RecordType.
2) To address the performance problems with #1, a cache bit was added to
CXXRecordDecl, and at the completion of every C++ class, the
RecordType was queried to determine if it was a standard layout
class, and that state was cached. Two things went very very wrong
with this. First, the caching version of the query *was never
called*. Even within the recursive steps of the walk over all fields
and bases the caching variant was not called, making each query
a full *recursive* walk. Second, despite the cache not being used, it
was computed for every class declared, even when the trait was never
used in the program. This probably significantly regressed compile
time performance for edge-case files.
3) An ASTContext was required merely to query the type trait because
querying it performed the actual computations.
4) The caching bit wasn't managed correctly (uninitialized).
The new implementation follows the system for all the other traits on
C++ classes by encoding all the state needed in the definition data and
building up the trait incrementally as each base and member are added to
the definition of the class.
The idiosyncracies of the specification of standard-layout classes
requires more state than I would like; currently 5 bits. I could
eliminate one of the bits easily at the expense of both clarity and
resilience of the code. I might be able to eliminate one of the other
bits by computing its state in terms of other state bits in the
definition. I've already done that in one place where there was a fairly
simple way to achieve it.
It's possible some of the bits could be moved out of the definition data
and into some other structure which isn't serialized if the serialized
bloat is a problem. That would preclude serialization of a partial class
declaration, but that's likely already precluded.
Comments on any of these issues welcome.
llvm-svn: 130601
2011-04-30 17:17:45 +08:00
|
|
|
// C++0x [class]p7:
|
|
|
|
// A standard-layout class is a class that: [...]
|
|
|
|
// -- has no virtual functions
|
2011-04-30 18:07:30 +08:00
|
|
|
data().IsStandardLayout = false;
|
2010-09-29 03:45:33 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2010-09-28 06:06:20 +08:00
|
|
|
if (D->isImplicit()) {
|
2010-10-25 01:26:54 +08:00
|
|
|
// Notify that an implicit member was added after the definition
|
|
|
|
// was completed.
|
|
|
|
if (!isBeingDefined())
|
|
|
|
if (ASTMutationListener *L = getASTMutationListener())
|
|
|
|
L->AddedCXXImplicitMember(data().Definition, D);
|
2010-10-21 07:48:42 +08:00
|
|
|
|
2011-05-26 04:50:04 +08:00
|
|
|
// If this is a special member function, note that it was added and then
|
|
|
|
// return early.
|
2010-09-28 06:06:20 +08:00
|
|
|
if (CXXConstructorDecl *Constructor = dyn_cast<CXXConstructorDecl>(D)) {
|
2011-12-22 10:22:31 +08:00
|
|
|
if (Constructor->isDefaultConstructor()) {
|
2010-09-28 06:06:20 +08:00
|
|
|
data().DeclaredDefaultConstructor = true;
|
2011-12-22 10:22:31 +08:00
|
|
|
if (Constructor->isConstexpr()) {
|
|
|
|
data().HasConstexprDefaultConstructor = true;
|
|
|
|
data().HasConstexprNonCopyMoveConstructor = true;
|
|
|
|
}
|
|
|
|
} else if (Constructor->isCopyConstructor()) {
|
2010-09-28 06:06:20 +08:00
|
|
|
data().DeclaredCopyConstructor = true;
|
2011-12-22 10:22:31 +08:00
|
|
|
if (Constructor->isConstexpr())
|
|
|
|
data().HasConstexprCopyConstructor = true;
|
|
|
|
} else if (Constructor->isMoveConstructor()) {
|
2011-05-26 04:50:04 +08:00
|
|
|
data().DeclaredMoveConstructor = true;
|
2011-12-22 10:22:31 +08:00
|
|
|
if (Constructor->isConstexpr())
|
|
|
|
data().HasConstexprMoveConstructor = true;
|
|
|
|
} else
|
2011-05-26 04:50:04 +08:00
|
|
|
goto NotASpecialMember;
|
2010-09-29 04:38:10 +08:00
|
|
|
return;
|
2011-05-26 04:50:04 +08:00
|
|
|
} else if (isa<CXXDestructorDecl>(D)) {
|
2010-09-28 06:48:58 +08:00
|
|
|
data().DeclaredDestructor = true;
|
2010-09-29 04:38:10 +08:00
|
|
|
return;
|
2011-05-26 04:50:04 +08:00
|
|
|
} else if (CXXMethodDecl *Method = dyn_cast<CXXMethodDecl>(D)) {
|
|
|
|
if (Method->isCopyAssignmentOperator())
|
2010-09-28 06:06:20 +08:00
|
|
|
data().DeclaredCopyAssignment = true;
|
2011-05-26 04:50:04 +08:00
|
|
|
else if (Method->isMoveAssignmentOperator())
|
|
|
|
data().DeclaredMoveAssignment = true;
|
|
|
|
else
|
|
|
|
goto NotASpecialMember;
|
2010-09-29 04:38:10 +08:00
|
|
|
return;
|
2010-09-28 06:06:20 +08:00
|
|
|
}
|
2010-09-29 04:38:10 +08:00
|
|
|
|
2011-05-26 04:50:04 +08:00
|
|
|
NotASpecialMember:;
|
2010-09-29 04:38:10 +08:00
|
|
|
// Any other implicit declarations are handled like normal declarations.
|
2010-09-28 05:17:54 +08:00
|
|
|
}
|
|
|
|
|
2010-09-28 06:06:20 +08:00
|
|
|
// Handle (user-declared) constructors.
|
|
|
|
if (CXXConstructorDecl *Constructor = dyn_cast<CXXConstructorDecl>(D)) {
|
|
|
|
// Note that we have a user-declared constructor.
|
|
|
|
data().UserDeclaredConstructor = true;
|
2009-06-18 06:44:31 +08:00
|
|
|
|
2011-09-05 10:13:09 +08:00
|
|
|
// Technically, "user-provided" is only defined for special member
|
|
|
|
// functions, but the intent of the standard is clearly that it should apply
|
|
|
|
// to all functions.
|
|
|
|
bool UserProvided = Constructor->isUserProvided();
|
2010-09-28 06:06:20 +08:00
|
|
|
|
2011-05-10 05:45:35 +08:00
|
|
|
if (Constructor->isDefaultConstructor()) {
|
|
|
|
data().DeclaredDefaultConstructor = true;
|
2011-09-05 10:13:09 +08:00
|
|
|
if (UserProvided) {
|
2011-12-22 10:22:31 +08:00
|
|
|
// C++0x [class.ctor]p5:
|
|
|
|
// A default constructor is trivial if it is not user-provided [...]
|
2011-05-10 05:45:35 +08:00
|
|
|
data().HasTrivialDefaultConstructor = false;
|
2011-05-12 06:34:38 +08:00
|
|
|
data().UserProvidedDefaultConstructor = true;
|
2011-05-10 05:45:35 +08:00
|
|
|
}
|
2011-12-22 10:22:31 +08:00
|
|
|
if (Constructor->isConstexpr()) {
|
|
|
|
data().HasConstexprDefaultConstructor = true;
|
|
|
|
data().HasConstexprNonCopyMoveConstructor = true;
|
|
|
|
}
|
2011-05-10 05:45:35 +08:00
|
|
|
}
|
2010-09-28 06:06:20 +08:00
|
|
|
|
2011-04-24 07:10:33 +08:00
|
|
|
// Note when we have a user-declared copy or move constructor, which will
|
|
|
|
// suppress the implicit declaration of those constructors.
|
|
|
|
if (!FunTmpl) {
|
|
|
|
if (Constructor->isCopyConstructor()) {
|
|
|
|
data().UserDeclaredCopyConstructor = true;
|
|
|
|
data().DeclaredCopyConstructor = true;
|
|
|
|
|
|
|
|
// C++0x [class.copy]p13:
|
2011-05-10 02:22:59 +08:00
|
|
|
// A copy/move constructor for class X is trivial if it is not
|
|
|
|
// user-provided [...]
|
2011-09-05 10:13:09 +08:00
|
|
|
if (UserProvided)
|
2011-05-10 02:22:59 +08:00
|
|
|
data().HasTrivialCopyConstructor = false;
|
2011-12-22 10:22:31 +08:00
|
|
|
|
|
|
|
if (Constructor->isConstexpr())
|
|
|
|
data().HasConstexprCopyConstructor = true;
|
2011-04-24 07:10:33 +08:00
|
|
|
} else if (Constructor->isMoveConstructor()) {
|
2011-05-26 04:50:04 +08:00
|
|
|
data().UserDeclaredMoveConstructor = true;
|
|
|
|
data().DeclaredMoveConstructor = true;
|
|
|
|
|
2011-04-24 07:10:33 +08:00
|
|
|
// C++0x [class.copy]p13:
|
2011-05-10 02:22:59 +08:00
|
|
|
// A copy/move constructor for class X is trivial if it is not
|
|
|
|
// user-provided [...]
|
2011-09-05 10:13:09 +08:00
|
|
|
if (UserProvided)
|
2011-05-10 02:22:59 +08:00
|
|
|
data().HasTrivialMoveConstructor = false;
|
2011-12-22 10:22:31 +08:00
|
|
|
|
|
|
|
if (Constructor->isConstexpr())
|
|
|
|
data().HasConstexprMoveConstructor = true;
|
2011-04-24 07:10:33 +08:00
|
|
|
}
|
2010-09-28 06:06:20 +08:00
|
|
|
}
|
2011-08-11 02:11:37 +08:00
|
|
|
if (Constructor->isConstexpr() && !Constructor->isCopyOrMoveConstructor()) {
|
|
|
|
// Record if we see any constexpr constructors which are neither copy
|
2011-04-24 10:49:34 +08:00
|
|
|
// nor move constructors.
|
2011-08-11 02:11:37 +08:00
|
|
|
data().HasConstexprNonCopyMoveConstructor = true;
|
2011-04-24 10:49:34 +08:00
|
|
|
}
|
2011-04-24 07:10:33 +08:00
|
|
|
|
2011-05-10 05:45:35 +08:00
|
|
|
// C++ [dcl.init.aggr]p1:
|
|
|
|
// An aggregate is an array or a class with no user-declared
|
|
|
|
// constructors [...].
|
|
|
|
// C++0x [dcl.init.aggr]p1:
|
|
|
|
// An aggregate is an array or a class with no user-provided
|
|
|
|
// constructors [...].
|
2012-03-11 15:00:24 +08:00
|
|
|
if (!getASTContext().getLangOpts().CPlusPlus0x || UserProvided)
|
2011-05-10 05:45:35 +08:00
|
|
|
data().Aggregate = false;
|
|
|
|
|
|
|
|
// C++ [class]p4:
|
|
|
|
// A POD-struct is an aggregate class [...]
|
|
|
|
// Since the POD bit is meant to be C++03 POD-ness, clear it even if the
|
|
|
|
// type is technically an aggregate in C++0x since it wouldn't be in 03.
|
|
|
|
data().PlainOldData = false;
|
|
|
|
|
2010-09-28 06:06:20 +08:00
|
|
|
return;
|
2009-07-23 02:25:24 +08:00
|
|
|
}
|
2008-11-01 04:25:05 +08:00
|
|
|
|
2010-09-28 06:48:58 +08:00
|
|
|
// Handle (user-declared) destructors.
|
2011-05-17 06:41:40 +08:00
|
|
|
if (CXXDestructorDecl *DD = dyn_cast<CXXDestructorDecl>(D)) {
|
2010-09-28 06:48:58 +08:00
|
|
|
data().DeclaredDestructor = true;
|
|
|
|
data().UserDeclaredDestructor = true;
|
2012-02-25 15:33:38 +08:00
|
|
|
data().HasIrrelevantDestructor = false;
|
|
|
|
|
2010-09-29 03:45:33 +08:00
|
|
|
// C++ [class]p4:
|
|
|
|
// A POD-struct is an aggregate class that has [...] no user-defined
|
|
|
|
// destructor.
|
2011-05-17 06:41:40 +08:00
|
|
|
// This bit is the C++03 POD bit, not the 0x one.
|
2010-09-29 03:45:33 +08:00
|
|
|
data().PlainOldData = false;
|
|
|
|
|
2011-11-08 04:56:01 +08:00
|
|
|
// C++11 [class.dtor]p5:
|
|
|
|
// A destructor is trivial if it is not user-provided and if
|
|
|
|
// -- the destructor is not virtual.
|
2011-12-22 10:22:31 +08:00
|
|
|
if (DD->isUserProvided() || DD->isVirtual()) {
|
2011-05-17 06:41:40 +08:00
|
|
|
data().HasTrivialDestructor = false;
|
2011-12-22 10:22:31 +08:00
|
|
|
// C++11 [dcl.constexpr]p1:
|
|
|
|
// The constexpr specifier shall be applied only to [...] the
|
|
|
|
// declaration of a static data member of a literal type.
|
|
|
|
// C++11 [basic.types]p10:
|
|
|
|
// A type is a literal type if it is [...] a class type that [...] has
|
|
|
|
// a trivial destructor.
|
|
|
|
data().DefaultedDefaultConstructorIsConstexpr = false;
|
|
|
|
data().DefaultedCopyConstructorIsConstexpr = false;
|
|
|
|
data().DefaultedMoveConstructorIsConstexpr = false;
|
|
|
|
}
|
2010-09-29 04:50:54 +08:00
|
|
|
|
2010-09-28 06:48:58 +08:00
|
|
|
return;
|
|
|
|
}
|
2009-10-14 07:45:19 +08:00
|
|
|
|
2010-09-28 06:48:58 +08:00
|
|
|
// Handle (user-declared) member functions.
|
2010-09-28 06:06:20 +08:00
|
|
|
if (CXXMethodDecl *Method = dyn_cast<CXXMethodDecl>(D)) {
|
2011-05-26 04:50:04 +08:00
|
|
|
if (Method->isCopyAssignmentOperator()) {
|
2010-09-28 06:06:20 +08:00
|
|
|
// C++ [class]p4:
|
2011-04-24 07:10:33 +08:00
|
|
|
// A POD-struct is an aggregate class that [...] has no user-defined
|
|
|
|
// copy assignment operator [...].
|
2011-05-17 06:41:40 +08:00
|
|
|
// This is the C++03 bit only.
|
2010-09-28 06:06:20 +08:00
|
|
|
data().PlainOldData = false;
|
2011-04-24 07:10:33 +08:00
|
|
|
|
2011-05-26 04:50:04 +08:00
|
|
|
// This is a copy assignment operator.
|
2011-04-24 07:10:33 +08:00
|
|
|
|
2011-05-26 04:50:04 +08:00
|
|
|
// Suppress the implicit declaration of a copy constructor.
|
|
|
|
data().UserDeclaredCopyAssignment = true;
|
|
|
|
data().DeclaredCopyAssignment = true;
|
2011-04-24 07:10:33 +08:00
|
|
|
|
2011-05-26 04:50:04 +08:00
|
|
|
// C++0x [class.copy]p27:
|
|
|
|
// A copy/move assignment operator for class X is trivial if it is
|
|
|
|
// neither user-provided nor deleted [...]
|
|
|
|
if (Method->isUserProvided())
|
|
|
|
data().HasTrivialCopyAssignment = false;
|
2011-04-24 07:10:33 +08:00
|
|
|
|
2011-05-26 04:50:04 +08:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (Method->isMoveAssignmentOperator()) {
|
|
|
|
// This is an extension in C++03 mode, but we'll keep consistency by
|
|
|
|
// taking a move assignment operator to induce non-POD-ness
|
|
|
|
data().PlainOldData = false;
|
|
|
|
|
|
|
|
// This is a move assignment operator.
|
|
|
|
data().UserDeclaredMoveAssignment = true;
|
|
|
|
data().DeclaredMoveAssignment = true;
|
|
|
|
|
|
|
|
// C++0x [class.copy]p27:
|
|
|
|
// A copy/move assignment operator for class X is trivial if it is
|
|
|
|
// neither user-provided nor deleted [...]
|
|
|
|
if (Method->isUserProvided())
|
|
|
|
data().HasTrivialMoveAssignment = false;
|
2010-09-28 06:06:20 +08:00
|
|
|
}
|
2011-04-24 07:10:33 +08:00
|
|
|
|
2010-09-29 12:25:11 +08:00
|
|
|
// Keep the list of conversion functions up-to-date.
|
|
|
|
if (CXXConversionDecl *Conversion = dyn_cast<CXXConversionDecl>(D)) {
|
|
|
|
// We don't record specializations.
|
|
|
|
if (Conversion->getPrimaryTemplate())
|
|
|
|
return;
|
|
|
|
|
|
|
|
// FIXME: We intentionally don't use the decl's access here because it
|
|
|
|
// hasn't been set yet. That's really just a misdesign in Sema.
|
|
|
|
|
|
|
|
if (FunTmpl) {
|
2012-01-15 00:38:05 +08:00
|
|
|
if (FunTmpl->getPreviousDecl())
|
|
|
|
data().Conversions.replace(FunTmpl->getPreviousDecl(),
|
2010-09-29 12:25:11 +08:00
|
|
|
FunTmpl);
|
|
|
|
else
|
|
|
|
data().Conversions.addDecl(FunTmpl);
|
|
|
|
} else {
|
2012-01-15 00:38:05 +08:00
|
|
|
if (Conversion->getPreviousDecl())
|
|
|
|
data().Conversions.replace(Conversion->getPreviousDecl(),
|
2010-09-29 12:25:11 +08:00
|
|
|
Conversion);
|
|
|
|
else
|
|
|
|
data().Conversions.addDecl(Conversion);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2009-01-06 04:52:13 +08:00
|
|
|
return;
|
2010-09-28 06:06:20 +08:00
|
|
|
}
|
2010-09-29 03:45:33 +08:00
|
|
|
|
|
|
|
// Handle non-static data members.
|
|
|
|
if (FieldDecl *Field = dyn_cast<FieldDecl>(D)) {
|
2011-10-11 01:22:13 +08:00
|
|
|
// C++ [class.bit]p2:
|
|
|
|
// A declaration for a bit-field that omits the identifier declares an
|
|
|
|
// unnamed bit-field. Unnamed bit-fields are not members and cannot be
|
|
|
|
// initialized.
|
|
|
|
if (Field->isUnnamedBitfield())
|
|
|
|
return;
|
|
|
|
|
2010-09-29 03:45:33 +08:00
|
|
|
// C++ [dcl.init.aggr]p1:
|
|
|
|
// An aggregate is an array or a class (clause 9) with [...] no
|
|
|
|
// private or protected non-static data members (clause 11).
|
|
|
|
//
|
|
|
|
// A POD must be an aggregate.
|
|
|
|
if (D->getAccess() == AS_private || D->getAccess() == AS_protected) {
|
|
|
|
data().Aggregate = false;
|
|
|
|
data().PlainOldData = false;
|
|
|
|
}
|
Completely re-implement the core logic behind the __is_standard_layout
type trait. The previous implementation suffered from several problems:
1) It implemented all of the logic in RecordType by walking over every
base and field in a CXXRecordDecl and validating the constraints of
the standard. This made for very straightforward code, but is
extremely inefficient. It also is conceptually wrong, the logic tied
to the C++ definition of standard-layout classes should be in
CXXRecordDecl, not RecordType.
2) To address the performance problems with #1, a cache bit was added to
CXXRecordDecl, and at the completion of every C++ class, the
RecordType was queried to determine if it was a standard layout
class, and that state was cached. Two things went very very wrong
with this. First, the caching version of the query *was never
called*. Even within the recursive steps of the walk over all fields
and bases the caching variant was not called, making each query
a full *recursive* walk. Second, despite the cache not being used, it
was computed for every class declared, even when the trait was never
used in the program. This probably significantly regressed compile
time performance for edge-case files.
3) An ASTContext was required merely to query the type trait because
querying it performed the actual computations.
4) The caching bit wasn't managed correctly (uninitialized).
The new implementation follows the system for all the other traits on
C++ classes by encoding all the state needed in the definition data and
building up the trait incrementally as each base and member are added to
the definition of the class.
The idiosyncracies of the specification of standard-layout classes
requires more state than I would like; currently 5 bits. I could
eliminate one of the bits easily at the expense of both clarity and
resilience of the code. I might be able to eliminate one of the other
bits by computing its state in terms of other state bits in the
definition. I've already done that in one place where there was a fairly
simple way to achieve it.
It's possible some of the bits could be moved out of the definition data
and into some other structure which isn't serialized if the serialized
bloat is a problem. That would preclude serialization of a partial class
declaration, but that's likely already precluded.
Comments on any of these issues welcome.
llvm-svn: 130601
2011-04-30 17:17:45 +08:00
|
|
|
|
|
|
|
// C++0x [class]p7:
|
|
|
|
// A standard-layout class is a class that:
|
|
|
|
// [...]
|
|
|
|
// -- has the same access control for all non-static data members,
|
|
|
|
switch (D->getAccess()) {
|
|
|
|
case AS_private: data().HasPrivateFields = true; break;
|
|
|
|
case AS_protected: data().HasProtectedFields = true; break;
|
|
|
|
case AS_public: data().HasPublicFields = true; break;
|
2011-09-23 13:06:16 +08:00
|
|
|
case AS_none: llvm_unreachable("Invalid access specifier");
|
Completely re-implement the core logic behind the __is_standard_layout
type trait. The previous implementation suffered from several problems:
1) It implemented all of the logic in RecordType by walking over every
base and field in a CXXRecordDecl and validating the constraints of
the standard. This made for very straightforward code, but is
extremely inefficient. It also is conceptually wrong, the logic tied
to the C++ definition of standard-layout classes should be in
CXXRecordDecl, not RecordType.
2) To address the performance problems with #1, a cache bit was added to
CXXRecordDecl, and at the completion of every C++ class, the
RecordType was queried to determine if it was a standard layout
class, and that state was cached. Two things went very very wrong
with this. First, the caching version of the query *was never
called*. Even within the recursive steps of the walk over all fields
and bases the caching variant was not called, making each query
a full *recursive* walk. Second, despite the cache not being used, it
was computed for every class declared, even when the trait was never
used in the program. This probably significantly regressed compile
time performance for edge-case files.
3) An ASTContext was required merely to query the type trait because
querying it performed the actual computations.
4) The caching bit wasn't managed correctly (uninitialized).
The new implementation follows the system for all the other traits on
C++ classes by encoding all the state needed in the definition data and
building up the trait incrementally as each base and member are added to
the definition of the class.
The idiosyncracies of the specification of standard-layout classes
requires more state than I would like; currently 5 bits. I could
eliminate one of the bits easily at the expense of both clarity and
resilience of the code. I might be able to eliminate one of the other
bits by computing its state in terms of other state bits in the
definition. I've already done that in one place where there was a fairly
simple way to achieve it.
It's possible some of the bits could be moved out of the definition data
and into some other structure which isn't serialized if the serialized
bloat is a problem. That would preclude serialization of a partial class
declaration, but that's likely already precluded.
Comments on any of these issues welcome.
llvm-svn: 130601
2011-04-30 17:17:45 +08:00
|
|
|
};
|
|
|
|
if ((data().HasPrivateFields + data().HasProtectedFields +
|
|
|
|
data().HasPublicFields) > 1)
|
2011-04-30 18:07:30 +08:00
|
|
|
data().IsStandardLayout = false;
|
Completely re-implement the core logic behind the __is_standard_layout
type trait. The previous implementation suffered from several problems:
1) It implemented all of the logic in RecordType by walking over every
base and field in a CXXRecordDecl and validating the constraints of
the standard. This made for very straightforward code, but is
extremely inefficient. It also is conceptually wrong, the logic tied
to the C++ definition of standard-layout classes should be in
CXXRecordDecl, not RecordType.
2) To address the performance problems with #1, a cache bit was added to
CXXRecordDecl, and at the completion of every C++ class, the
RecordType was queried to determine if it was a standard layout
class, and that state was cached. Two things went very very wrong
with this. First, the caching version of the query *was never
called*. Even within the recursive steps of the walk over all fields
and bases the caching variant was not called, making each query
a full *recursive* walk. Second, despite the cache not being used, it
was computed for every class declared, even when the trait was never
used in the program. This probably significantly regressed compile
time performance for edge-case files.
3) An ASTContext was required merely to query the type trait because
querying it performed the actual computations.
4) The caching bit wasn't managed correctly (uninitialized).
The new implementation follows the system for all the other traits on
C++ classes by encoding all the state needed in the definition data and
building up the trait incrementally as each base and member are added to
the definition of the class.
The idiosyncracies of the specification of standard-layout classes
requires more state than I would like; currently 5 bits. I could
eliminate one of the bits easily at the expense of both clarity and
resilience of the code. I might be able to eliminate one of the other
bits by computing its state in terms of other state bits in the
definition. I've already done that in one place where there was a fairly
simple way to achieve it.
It's possible some of the bits could be moved out of the definition data
and into some other structure which isn't serialized if the serialized
bloat is a problem. That would preclude serialization of a partial class
declaration, but that's likely already precluded.
Comments on any of these issues welcome.
llvm-svn: 130601
2011-04-30 17:17:45 +08:00
|
|
|
|
2011-05-13 09:05:07 +08:00
|
|
|
// Keep track of the presence of mutable fields.
|
|
|
|
if (Field->isMutable())
|
|
|
|
data().HasMutableFields = true;
|
|
|
|
|
2011-04-24 07:10:33 +08:00
|
|
|
// C++0x [class]p9:
|
2010-09-29 03:45:33 +08:00
|
|
|
// A POD struct is a class that is both a trivial class and a
|
|
|
|
// standard-layout class, and has no non-static data members of type
|
|
|
|
// non-POD struct, non-POD union (or array of such types).
|
2011-06-16 07:02:42 +08:00
|
|
|
//
|
|
|
|
// Automatic Reference Counting: the presence of a member of Objective-C pointer type
|
|
|
|
// that does not explicitly have no lifetime makes the class a non-POD.
|
|
|
|
// However, we delay setting PlainOldData to false in this case so that
|
|
|
|
// Sema has a chance to diagnostic causes where the same class will be
|
|
|
|
// non-POD with Automatic Reference Counting but a POD without Instant Objects.
|
|
|
|
// In this case, the class will become a non-POD class when we complete
|
|
|
|
// the definition.
|
2010-09-29 03:45:33 +08:00
|
|
|
ASTContext &Context = getASTContext();
|
|
|
|
QualType T = Context.getBaseElementType(Field->getType());
|
2011-06-16 07:02:42 +08:00
|
|
|
if (T->isObjCRetainableType() || T.isObjCGCStrong()) {
|
2012-03-11 15:00:24 +08:00
|
|
|
if (!Context.getLangOpts().ObjCAutoRefCount ||
|
2011-06-16 07:02:42 +08:00
|
|
|
T.getObjCLifetime() != Qualifiers::OCL_ExplicitNone)
|
|
|
|
setHasObjectMember(true);
|
|
|
|
} else if (!T.isPODType(Context))
|
2010-09-29 03:45:33 +08:00
|
|
|
data().PlainOldData = false;
|
2011-06-16 07:02:42 +08:00
|
|
|
|
Completely re-implement the core logic behind the __is_standard_layout
type trait. The previous implementation suffered from several problems:
1) It implemented all of the logic in RecordType by walking over every
base and field in a CXXRecordDecl and validating the constraints of
the standard. This made for very straightforward code, but is
extremely inefficient. It also is conceptually wrong, the logic tied
to the C++ definition of standard-layout classes should be in
CXXRecordDecl, not RecordType.
2) To address the performance problems with #1, a cache bit was added to
CXXRecordDecl, and at the completion of every C++ class, the
RecordType was queried to determine if it was a standard layout
class, and that state was cached. Two things went very very wrong
with this. First, the caching version of the query *was never
called*. Even within the recursive steps of the walk over all fields
and bases the caching variant was not called, making each query
a full *recursive* walk. Second, despite the cache not being used, it
was computed for every class declared, even when the trait was never
used in the program. This probably significantly regressed compile
time performance for edge-case files.
3) An ASTContext was required merely to query the type trait because
querying it performed the actual computations.
4) The caching bit wasn't managed correctly (uninitialized).
The new implementation follows the system for all the other traits on
C++ classes by encoding all the state needed in the definition data and
building up the trait incrementally as each base and member are added to
the definition of the class.
The idiosyncracies of the specification of standard-layout classes
requires more state than I would like; currently 5 bits. I could
eliminate one of the bits easily at the expense of both clarity and
resilience of the code. I might be able to eliminate one of the other
bits by computing its state in terms of other state bits in the
definition. I've already done that in one place where there was a fairly
simple way to achieve it.
It's possible some of the bits could be moved out of the definition data
and into some other structure which isn't serialized if the serialized
bloat is a problem. That would preclude serialization of a partial class
declaration, but that's likely already precluded.
Comments on any of these issues welcome.
llvm-svn: 130601
2011-04-30 17:17:45 +08:00
|
|
|
if (T->isReferenceType()) {
|
2011-05-10 02:22:59 +08:00
|
|
|
data().HasTrivialDefaultConstructor = false;
|
2011-04-24 10:49:34 +08:00
|
|
|
|
Completely re-implement the core logic behind the __is_standard_layout
type trait. The previous implementation suffered from several problems:
1) It implemented all of the logic in RecordType by walking over every
base and field in a CXXRecordDecl and validating the constraints of
the standard. This made for very straightforward code, but is
extremely inefficient. It also is conceptually wrong, the logic tied
to the C++ definition of standard-layout classes should be in
CXXRecordDecl, not RecordType.
2) To address the performance problems with #1, a cache bit was added to
CXXRecordDecl, and at the completion of every C++ class, the
RecordType was queried to determine if it was a standard layout
class, and that state was cached. Two things went very very wrong
with this. First, the caching version of the query *was never
called*. Even within the recursive steps of the walk over all fields
and bases the caching variant was not called, making each query
a full *recursive* walk. Second, despite the cache not being used, it
was computed for every class declared, even when the trait was never
used in the program. This probably significantly regressed compile
time performance for edge-case files.
3) An ASTContext was required merely to query the type trait because
querying it performed the actual computations.
4) The caching bit wasn't managed correctly (uninitialized).
The new implementation follows the system for all the other traits on
C++ classes by encoding all the state needed in the definition data and
building up the trait incrementally as each base and member are added to
the definition of the class.
The idiosyncracies of the specification of standard-layout classes
requires more state than I would like; currently 5 bits. I could
eliminate one of the bits easily at the expense of both clarity and
resilience of the code. I might be able to eliminate one of the other
bits by computing its state in terms of other state bits in the
definition. I've already done that in one place where there was a fairly
simple way to achieve it.
It's possible some of the bits could be moved out of the definition data
and into some other structure which isn't serialized if the serialized
bloat is a problem. That would preclude serialization of a partial class
declaration, but that's likely already precluded.
Comments on any of these issues welcome.
llvm-svn: 130601
2011-04-30 17:17:45 +08:00
|
|
|
// C++0x [class]p7:
|
|
|
|
// A standard-layout class is a class that:
|
|
|
|
// -- has no non-static data members of type [...] reference,
|
2011-04-30 18:07:30 +08:00
|
|
|
data().IsStandardLayout = false;
|
Completely re-implement the core logic behind the __is_standard_layout
type trait. The previous implementation suffered from several problems:
1) It implemented all of the logic in RecordType by walking over every
base and field in a CXXRecordDecl and validating the constraints of
the standard. This made for very straightforward code, but is
extremely inefficient. It also is conceptually wrong, the logic tied
to the C++ definition of standard-layout classes should be in
CXXRecordDecl, not RecordType.
2) To address the performance problems with #1, a cache bit was added to
CXXRecordDecl, and at the completion of every C++ class, the
RecordType was queried to determine if it was a standard layout
class, and that state was cached. Two things went very very wrong
with this. First, the caching version of the query *was never
called*. Even within the recursive steps of the walk over all fields
and bases the caching variant was not called, making each query
a full *recursive* walk. Second, despite the cache not being used, it
was computed for every class declared, even when the trait was never
used in the program. This probably significantly regressed compile
time performance for edge-case files.
3) An ASTContext was required merely to query the type trait because
querying it performed the actual computations.
4) The caching bit wasn't managed correctly (uninitialized).
The new implementation follows the system for all the other traits on
C++ classes by encoding all the state needed in the definition data and
building up the trait incrementally as each base and member are added to
the definition of the class.
The idiosyncracies of the specification of standard-layout classes
requires more state than I would like; currently 5 bits. I could
eliminate one of the bits easily at the expense of both clarity and
resilience of the code. I might be able to eliminate one of the other
bits by computing its state in terms of other state bits in the
definition. I've already done that in one place where there was a fairly
simple way to achieve it.
It's possible some of the bits could be moved out of the definition data
and into some other structure which isn't serialized if the serialized
bloat is a problem. That would preclude serialization of a partial class
declaration, but that's likely already precluded.
Comments on any of these issues welcome.
llvm-svn: 130601
2011-04-30 17:17:45 +08:00
|
|
|
}
|
|
|
|
|
2012-02-13 11:54:03 +08:00
|
|
|
// Record if this field is the first non-literal or volatile field or base.
|
|
|
|
if (!T->isLiteralType() || T.isVolatileQualified())
|
2011-04-24 10:49:34 +08:00
|
|
|
data().HasNonLiteralTypeFieldsOrBases = true;
|
|
|
|
|
2011-06-12 01:19:42 +08:00
|
|
|
if (Field->hasInClassInitializer()) {
|
|
|
|
// C++0x [class]p5:
|
|
|
|
// A default constructor is trivial if [...] no non-static data member
|
|
|
|
// of its class has a brace-or-equal-initializer.
|
|
|
|
data().HasTrivialDefaultConstructor = false;
|
|
|
|
|
|
|
|
// C++0x [dcl.init.aggr]p1:
|
|
|
|
// An aggregate is a [...] class with [...] no
|
|
|
|
// brace-or-equal-initializers for non-static data members.
|
|
|
|
data().Aggregate = false;
|
|
|
|
|
|
|
|
// C++0x [class]p10:
|
|
|
|
// A POD struct is [...] a trivial class.
|
|
|
|
data().PlainOldData = false;
|
|
|
|
}
|
|
|
|
|
2010-09-29 04:50:54 +08:00
|
|
|
if (const RecordType *RecordTy = T->getAs<RecordType>()) {
|
|
|
|
CXXRecordDecl* FieldRec = cast<CXXRecordDecl>(RecordTy->getDecl());
|
|
|
|
if (FieldRec->getDefinition()) {
|
2011-05-10 02:22:59 +08:00
|
|
|
// C++0x [class.ctor]p5:
|
2011-12-22 10:22:31 +08:00
|
|
|
// A default constructor is trivial [...] if:
|
2011-05-10 02:22:59 +08:00
|
|
|
// -- for all the non-static data members of its class that are of
|
|
|
|
// class type (or array thereof), each such class has a trivial
|
|
|
|
// default constructor.
|
|
|
|
if (!FieldRec->hasTrivialDefaultConstructor())
|
|
|
|
data().HasTrivialDefaultConstructor = false;
|
2011-04-24 07:10:33 +08:00
|
|
|
|
|
|
|
// C++0x [class.copy]p13:
|
|
|
|
// A copy/move constructor for class X is trivial if [...]
|
|
|
|
// [...]
|
|
|
|
// -- for each non-static data member of X that is of class type (or
|
|
|
|
// an array thereof), the constructor selected to copy/move that
|
|
|
|
// member is trivial;
|
|
|
|
// FIXME: C++0x: We don't correctly model 'selected' constructors.
|
2010-09-29 04:50:54 +08:00
|
|
|
if (!FieldRec->hasTrivialCopyConstructor())
|
|
|
|
data().HasTrivialCopyConstructor = false;
|
2011-04-24 07:10:33 +08:00
|
|
|
if (!FieldRec->hasTrivialMoveConstructor())
|
|
|
|
data().HasTrivialMoveConstructor = false;
|
|
|
|
|
|
|
|
// C++0x [class.copy]p27:
|
|
|
|
// A copy/move assignment operator for class X is trivial if [...]
|
|
|
|
// [...]
|
|
|
|
// -- for each non-static data member of X that is of class type (or
|
|
|
|
// an array thereof), the assignment operator selected to
|
|
|
|
// copy/move that member is trivial;
|
|
|
|
// FIXME: C++0x: We don't correctly model 'selected' operators.
|
2010-09-29 04:50:54 +08:00
|
|
|
if (!FieldRec->hasTrivialCopyAssignment())
|
|
|
|
data().HasTrivialCopyAssignment = false;
|
2011-04-24 07:10:33 +08:00
|
|
|
if (!FieldRec->hasTrivialMoveAssignment())
|
|
|
|
data().HasTrivialMoveAssignment = false;
|
|
|
|
|
2010-09-29 04:50:54 +08:00
|
|
|
if (!FieldRec->hasTrivialDestructor())
|
|
|
|
data().HasTrivialDestructor = false;
|
2012-02-25 15:33:38 +08:00
|
|
|
if (!FieldRec->hasIrrelevantDestructor())
|
|
|
|
data().HasIrrelevantDestructor = false;
|
2011-06-16 07:02:42 +08:00
|
|
|
if (FieldRec->hasObjectMember())
|
|
|
|
setHasObjectMember(true);
|
Completely re-implement the core logic behind the __is_standard_layout
type trait. The previous implementation suffered from several problems:
1) It implemented all of the logic in RecordType by walking over every
base and field in a CXXRecordDecl and validating the constraints of
the standard. This made for very straightforward code, but is
extremely inefficient. It also is conceptually wrong, the logic tied
to the C++ definition of standard-layout classes should be in
CXXRecordDecl, not RecordType.
2) To address the performance problems with #1, a cache bit was added to
CXXRecordDecl, and at the completion of every C++ class, the
RecordType was queried to determine if it was a standard layout
class, and that state was cached. Two things went very very wrong
with this. First, the caching version of the query *was never
called*. Even within the recursive steps of the walk over all fields
and bases the caching variant was not called, making each query
a full *recursive* walk. Second, despite the cache not being used, it
was computed for every class declared, even when the trait was never
used in the program. This probably significantly regressed compile
time performance for edge-case files.
3) An ASTContext was required merely to query the type trait because
querying it performed the actual computations.
4) The caching bit wasn't managed correctly (uninitialized).
The new implementation follows the system for all the other traits on
C++ classes by encoding all the state needed in the definition data and
building up the trait incrementally as each base and member are added to
the definition of the class.
The idiosyncracies of the specification of standard-layout classes
requires more state than I would like; currently 5 bits. I could
eliminate one of the bits easily at the expense of both clarity and
resilience of the code. I might be able to eliminate one of the other
bits by computing its state in terms of other state bits in the
definition. I've already done that in one place where there was a fairly
simple way to achieve it.
It's possible some of the bits could be moved out of the definition data
and into some other structure which isn't serialized if the serialized
bloat is a problem. That would preclude serialization of a partial class
declaration, but that's likely already precluded.
Comments on any of these issues welcome.
llvm-svn: 130601
2011-04-30 17:17:45 +08:00
|
|
|
|
|
|
|
// C++0x [class]p7:
|
|
|
|
// A standard-layout class is a class that:
|
|
|
|
// -- has no non-static data members of type non-standard-layout
|
|
|
|
// class (or array of such types) [...]
|
2011-04-30 18:07:30 +08:00
|
|
|
if (!FieldRec->isStandardLayout())
|
|
|
|
data().IsStandardLayout = false;
|
Completely re-implement the core logic behind the __is_standard_layout
type trait. The previous implementation suffered from several problems:
1) It implemented all of the logic in RecordType by walking over every
base and field in a CXXRecordDecl and validating the constraints of
the standard. This made for very straightforward code, but is
extremely inefficient. It also is conceptually wrong, the logic tied
to the C++ definition of standard-layout classes should be in
CXXRecordDecl, not RecordType.
2) To address the performance problems with #1, a cache bit was added to
CXXRecordDecl, and at the completion of every C++ class, the
RecordType was queried to determine if it was a standard layout
class, and that state was cached. Two things went very very wrong
with this. First, the caching version of the query *was never
called*. Even within the recursive steps of the walk over all fields
and bases the caching variant was not called, making each query
a full *recursive* walk. Second, despite the cache not being used, it
was computed for every class declared, even when the trait was never
used in the program. This probably significantly regressed compile
time performance for edge-case files.
3) An ASTContext was required merely to query the type trait because
querying it performed the actual computations.
4) The caching bit wasn't managed correctly (uninitialized).
The new implementation follows the system for all the other traits on
C++ classes by encoding all the state needed in the definition data and
building up the trait incrementally as each base and member are added to
the definition of the class.
The idiosyncracies of the specification of standard-layout classes
requires more state than I would like; currently 5 bits. I could
eliminate one of the bits easily at the expense of both clarity and
resilience of the code. I might be able to eliminate one of the other
bits by computing its state in terms of other state bits in the
definition. I've already done that in one place where there was a fairly
simple way to achieve it.
It's possible some of the bits could be moved out of the definition data
and into some other structure which isn't serialized if the serialized
bloat is a problem. That would preclude serialization of a partial class
declaration, but that's likely already precluded.
Comments on any of these issues welcome.
llvm-svn: 130601
2011-04-30 17:17:45 +08:00
|
|
|
|
|
|
|
// C++0x [class]p7:
|
|
|
|
// A standard-layout class is a class that:
|
|
|
|
// [...]
|
|
|
|
// -- has no base classes of the same type as the first non-static
|
|
|
|
// data member.
|
|
|
|
// We don't want to expend bits in the state of the record decl
|
|
|
|
// tracking whether this is the first non-static data member so we
|
|
|
|
// cheat a bit and use some of the existing state: the empty bit.
|
|
|
|
// Virtual bases and virtual methods make a class non-empty, but they
|
|
|
|
// also make it non-standard-layout so we needn't check here.
|
|
|
|
// A non-empty base class may leave the class standard-layout, but not
|
|
|
|
// if we have arrived here, and have at least on non-static data
|
2011-04-30 18:07:30 +08:00
|
|
|
// member. If IsStandardLayout remains true, then the first non-static
|
Completely re-implement the core logic behind the __is_standard_layout
type trait. The previous implementation suffered from several problems:
1) It implemented all of the logic in RecordType by walking over every
base and field in a CXXRecordDecl and validating the constraints of
the standard. This made for very straightforward code, but is
extremely inefficient. It also is conceptually wrong, the logic tied
to the C++ definition of standard-layout classes should be in
CXXRecordDecl, not RecordType.
2) To address the performance problems with #1, a cache bit was added to
CXXRecordDecl, and at the completion of every C++ class, the
RecordType was queried to determine if it was a standard layout
class, and that state was cached. Two things went very very wrong
with this. First, the caching version of the query *was never
called*. Even within the recursive steps of the walk over all fields
and bases the caching variant was not called, making each query
a full *recursive* walk. Second, despite the cache not being used, it
was computed for every class declared, even when the trait was never
used in the program. This probably significantly regressed compile
time performance for edge-case files.
3) An ASTContext was required merely to query the type trait because
querying it performed the actual computations.
4) The caching bit wasn't managed correctly (uninitialized).
The new implementation follows the system for all the other traits on
C++ classes by encoding all the state needed in the definition data and
building up the trait incrementally as each base and member are added to
the definition of the class.
The idiosyncracies of the specification of standard-layout classes
requires more state than I would like; currently 5 bits. I could
eliminate one of the bits easily at the expense of both clarity and
resilience of the code. I might be able to eliminate one of the other
bits by computing its state in terms of other state bits in the
definition. I've already done that in one place where there was a fairly
simple way to achieve it.
It's possible some of the bits could be moved out of the definition data
and into some other structure which isn't serialized if the serialized
bloat is a problem. That would preclude serialization of a partial class
declaration, but that's likely already precluded.
Comments on any of these issues welcome.
llvm-svn: 130601
2011-04-30 17:17:45 +08:00
|
|
|
// data member must come through here with Empty still true, and Empty
|
|
|
|
// will subsequently be set to false below.
|
2011-04-30 18:07:30 +08:00
|
|
|
if (data().IsStandardLayout && data().Empty) {
|
Completely re-implement the core logic behind the __is_standard_layout
type trait. The previous implementation suffered from several problems:
1) It implemented all of the logic in RecordType by walking over every
base and field in a CXXRecordDecl and validating the constraints of
the standard. This made for very straightforward code, but is
extremely inefficient. It also is conceptually wrong, the logic tied
to the C++ definition of standard-layout classes should be in
CXXRecordDecl, not RecordType.
2) To address the performance problems with #1, a cache bit was added to
CXXRecordDecl, and at the completion of every C++ class, the
RecordType was queried to determine if it was a standard layout
class, and that state was cached. Two things went very very wrong
with this. First, the caching version of the query *was never
called*. Even within the recursive steps of the walk over all fields
and bases the caching variant was not called, making each query
a full *recursive* walk. Second, despite the cache not being used, it
was computed for every class declared, even when the trait was never
used in the program. This probably significantly regressed compile
time performance for edge-case files.
3) An ASTContext was required merely to query the type trait because
querying it performed the actual computations.
4) The caching bit wasn't managed correctly (uninitialized).
The new implementation follows the system for all the other traits on
C++ classes by encoding all the state needed in the definition data and
building up the trait incrementally as each base and member are added to
the definition of the class.
The idiosyncracies of the specification of standard-layout classes
requires more state than I would like; currently 5 bits. I could
eliminate one of the bits easily at the expense of both clarity and
resilience of the code. I might be able to eliminate one of the other
bits by computing its state in terms of other state bits in the
definition. I've already done that in one place where there was a fairly
simple way to achieve it.
It's possible some of the bits could be moved out of the definition data
and into some other structure which isn't serialized if the serialized
bloat is a problem. That would preclude serialization of a partial class
declaration, but that's likely already precluded.
Comments on any of these issues welcome.
llvm-svn: 130601
2011-04-30 17:17:45 +08:00
|
|
|
for (CXXRecordDecl::base_class_const_iterator BI = bases_begin(),
|
|
|
|
BE = bases_end();
|
|
|
|
BI != BE; ++BI) {
|
|
|
|
if (Context.hasSameUnqualifiedType(BI->getType(), T)) {
|
2011-04-30 18:07:30 +08:00
|
|
|
data().IsStandardLayout = false;
|
Completely re-implement the core logic behind the __is_standard_layout
type trait. The previous implementation suffered from several problems:
1) It implemented all of the logic in RecordType by walking over every
base and field in a CXXRecordDecl and validating the constraints of
the standard. This made for very straightforward code, but is
extremely inefficient. It also is conceptually wrong, the logic tied
to the C++ definition of standard-layout classes should be in
CXXRecordDecl, not RecordType.
2) To address the performance problems with #1, a cache bit was added to
CXXRecordDecl, and at the completion of every C++ class, the
RecordType was queried to determine if it was a standard layout
class, and that state was cached. Two things went very very wrong
with this. First, the caching version of the query *was never
called*. Even within the recursive steps of the walk over all fields
and bases the caching variant was not called, making each query
a full *recursive* walk. Second, despite the cache not being used, it
was computed for every class declared, even when the trait was never
used in the program. This probably significantly regressed compile
time performance for edge-case files.
3) An ASTContext was required merely to query the type trait because
querying it performed the actual computations.
4) The caching bit wasn't managed correctly (uninitialized).
The new implementation follows the system for all the other traits on
C++ classes by encoding all the state needed in the definition data and
building up the trait incrementally as each base and member are added to
the definition of the class.
The idiosyncracies of the specification of standard-layout classes
requires more state than I would like; currently 5 bits. I could
eliminate one of the bits easily at the expense of both clarity and
resilience of the code. I might be able to eliminate one of the other
bits by computing its state in terms of other state bits in the
definition. I've already done that in one place where there was a fairly
simple way to achieve it.
It's possible some of the bits could be moved out of the definition data
and into some other structure which isn't serialized if the serialized
bloat is a problem. That would preclude serialization of a partial class
declaration, but that's likely already precluded.
Comments on any of these issues welcome.
llvm-svn: 130601
2011-04-30 17:17:45 +08:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2011-05-13 09:05:07 +08:00
|
|
|
|
|
|
|
// Keep track of the presence of mutable fields.
|
|
|
|
if (FieldRec->hasMutableFields())
|
|
|
|
data().HasMutableFields = true;
|
2011-12-22 10:22:31 +08:00
|
|
|
|
|
|
|
// C++11 [class.copy]p13:
|
|
|
|
// If the implicitly-defined constructor would satisfy the
|
|
|
|
// requirements of a constexpr constructor, the implicitly-defined
|
|
|
|
// constructor is constexpr.
|
|
|
|
// C++11 [dcl.constexpr]p4:
|
|
|
|
// -- every constructor involved in initializing non-static data
|
|
|
|
// members [...] shall be a constexpr constructor
|
|
|
|
if (!Field->hasInClassInitializer() &&
|
|
|
|
!FieldRec->hasConstexprDefaultConstructor())
|
|
|
|
// The standard requires any in-class initializer to be a constant
|
|
|
|
// expression. We consider this to be a defect.
|
|
|
|
data().DefaultedDefaultConstructorIsConstexpr = false;
|
|
|
|
|
|
|
|
if (!FieldRec->hasConstexprCopyConstructor())
|
|
|
|
data().DefaultedCopyConstructorIsConstexpr = false;
|
|
|
|
|
|
|
|
if (FieldRec->hasDeclaredMoveConstructor() ||
|
|
|
|
FieldRec->needsImplicitMoveConstructor())
|
|
|
|
// FIXME: If the implicit move constructor generated for the member's
|
|
|
|
// class would be ill-formed, the implicit move constructor generated
|
|
|
|
// for this class calls the member's copy constructor.
|
|
|
|
data().DefaultedMoveConstructorIsConstexpr &=
|
|
|
|
FieldRec->hasConstexprMoveConstructor();
|
|
|
|
else if (!FieldRec->hasConstexprCopyConstructor())
|
|
|
|
data().DefaultedMoveConstructorIsConstexpr = false;
|
2010-09-29 04:50:54 +08:00
|
|
|
}
|
2011-12-22 10:22:31 +08:00
|
|
|
} else {
|
|
|
|
// Base element type of field is a non-class type.
|
|
|
|
if (!T->isLiteralType()) {
|
|
|
|
data().DefaultedDefaultConstructorIsConstexpr = false;
|
|
|
|
data().DefaultedCopyConstructorIsConstexpr = false;
|
|
|
|
data().DefaultedMoveConstructorIsConstexpr = false;
|
|
|
|
} else if (!Field->hasInClassInitializer())
|
|
|
|
data().DefaultedDefaultConstructorIsConstexpr = false;
|
2010-09-29 04:50:54 +08:00
|
|
|
}
|
Completely re-implement the core logic behind the __is_standard_layout
type trait. The previous implementation suffered from several problems:
1) It implemented all of the logic in RecordType by walking over every
base and field in a CXXRecordDecl and validating the constraints of
the standard. This made for very straightforward code, but is
extremely inefficient. It also is conceptually wrong, the logic tied
to the C++ definition of standard-layout classes should be in
CXXRecordDecl, not RecordType.
2) To address the performance problems with #1, a cache bit was added to
CXXRecordDecl, and at the completion of every C++ class, the
RecordType was queried to determine if it was a standard layout
class, and that state was cached. Two things went very very wrong
with this. First, the caching version of the query *was never
called*. Even within the recursive steps of the walk over all fields
and bases the caching variant was not called, making each query
a full *recursive* walk. Second, despite the cache not being used, it
was computed for every class declared, even when the trait was never
used in the program. This probably significantly regressed compile
time performance for edge-case files.
3) An ASTContext was required merely to query the type trait because
querying it performed the actual computations.
4) The caching bit wasn't managed correctly (uninitialized).
The new implementation follows the system for all the other traits on
C++ classes by encoding all the state needed in the definition data and
building up the trait incrementally as each base and member are added to
the definition of the class.
The idiosyncracies of the specification of standard-layout classes
requires more state than I would like; currently 5 bits. I could
eliminate one of the bits easily at the expense of both clarity and
resilience of the code. I might be able to eliminate one of the other
bits by computing its state in terms of other state bits in the
definition. I've already done that in one place where there was a fairly
simple way to achieve it.
It's possible some of the bits could be moved out of the definition data
and into some other structure which isn't serialized if the serialized
bloat is a problem. That would preclude serialization of a partial class
declaration, but that's likely already precluded.
Comments on any of these issues welcome.
llvm-svn: 130601
2011-04-30 17:17:45 +08:00
|
|
|
|
|
|
|
// C++0x [class]p7:
|
|
|
|
// A standard-layout class is a class that:
|
|
|
|
// [...]
|
|
|
|
// -- either has no non-static data members in the most derived
|
|
|
|
// class and at most one base class with non-static data members,
|
|
|
|
// or has no base classes with non-static data members, and
|
|
|
|
// At this point we know that we have a non-static data member, so the last
|
|
|
|
// clause holds.
|
|
|
|
if (!data().HasNoNonEmptyBases)
|
2011-04-30 18:07:30 +08:00
|
|
|
data().IsStandardLayout = false;
|
Completely re-implement the core logic behind the __is_standard_layout
type trait. The previous implementation suffered from several problems:
1) It implemented all of the logic in RecordType by walking over every
base and field in a CXXRecordDecl and validating the constraints of
the standard. This made for very straightforward code, but is
extremely inefficient. It also is conceptually wrong, the logic tied
to the C++ definition of standard-layout classes should be in
CXXRecordDecl, not RecordType.
2) To address the performance problems with #1, a cache bit was added to
CXXRecordDecl, and at the completion of every C++ class, the
RecordType was queried to determine if it was a standard layout
class, and that state was cached. Two things went very very wrong
with this. First, the caching version of the query *was never
called*. Even within the recursive steps of the walk over all fields
and bases the caching variant was not called, making each query
a full *recursive* walk. Second, despite the cache not being used, it
was computed for every class declared, even when the trait was never
used in the program. This probably significantly regressed compile
time performance for edge-case files.
3) An ASTContext was required merely to query the type trait because
querying it performed the actual computations.
4) The caching bit wasn't managed correctly (uninitialized).
The new implementation follows the system for all the other traits on
C++ classes by encoding all the state needed in the definition data and
building up the trait incrementally as each base and member are added to
the definition of the class.
The idiosyncracies of the specification of standard-layout classes
requires more state than I would like; currently 5 bits. I could
eliminate one of the bits easily at the expense of both clarity and
resilience of the code. I might be able to eliminate one of the other
bits by computing its state in terms of other state bits in the
definition. I've already done that in one place where there was a fairly
simple way to achieve it.
It's possible some of the bits could be moved out of the definition data
and into some other structure which isn't serialized if the serialized
bloat is a problem. That would preclude serialization of a partial class
declaration, but that's likely already precluded.
Comments on any of these issues welcome.
llvm-svn: 130601
2011-04-30 17:17:45 +08:00
|
|
|
|
2010-09-29 04:38:10 +08:00
|
|
|
// If this is not a zero-length bit-field, then the class is not empty.
|
|
|
|
if (data().Empty) {
|
2011-10-11 02:28:20 +08:00
|
|
|
if (!Field->isBitField() ||
|
|
|
|
(!Field->getBitWidth()->isTypeDependent() &&
|
|
|
|
!Field->getBitWidth()->isValueDependent() &&
|
|
|
|
Field->getBitWidthValue(Context) != 0))
|
2010-09-29 04:38:10 +08:00
|
|
|
data().Empty = false;
|
|
|
|
}
|
2010-09-29 03:45:33 +08:00
|
|
|
}
|
2010-09-29 12:25:11 +08:00
|
|
|
|
|
|
|
// Handle using declarations of conversion functions.
|
|
|
|
if (UsingShadowDecl *Shadow = dyn_cast<UsingShadowDecl>(D))
|
|
|
|
if (Shadow->getDeclName().getNameKind()
|
|
|
|
== DeclarationName::CXXConversionFunctionName)
|
|
|
|
data().Conversions.addDecl(Shadow, Shadow->getAccess());
|
2009-01-06 04:52:13 +08:00
|
|
|
}
|
|
|
|
|
2012-01-24 00:58:45 +08:00
|
|
|
bool CXXRecordDecl::isCLike() const {
|
|
|
|
if (getTagKind() == TTK_Class || !TemplateOrInstantiation.isNull())
|
|
|
|
return false;
|
|
|
|
if (!hasDefinition())
|
|
|
|
return true;
|
|
|
|
|
2012-02-01 14:36:44 +08:00
|
|
|
return isPOD() && data().HasOnlyCMembers;
|
2012-01-24 00:58:45 +08:00
|
|
|
}
|
|
|
|
|
2012-02-10 15:45:31 +08:00
|
|
|
void CXXRecordDecl::getCaptureFields(
|
|
|
|
llvm::DenseMap<const VarDecl *, FieldDecl *> &Captures,
|
2012-02-11 08:18:00 +08:00
|
|
|
FieldDecl *&ThisCapture) const {
|
2012-02-10 15:45:31 +08:00
|
|
|
Captures.clear();
|
|
|
|
ThisCapture = 0;
|
|
|
|
|
2012-02-13 23:44:47 +08:00
|
|
|
LambdaDefinitionData &Lambda = getLambdaData();
|
2012-02-10 15:45:31 +08:00
|
|
|
RecordDecl::field_iterator Field = field_begin();
|
2012-02-14 01:20:40 +08:00
|
|
|
for (LambdaExpr::Capture *C = Lambda.Captures, *CEnd = C + Lambda.NumCaptures;
|
2012-02-10 15:45:31 +08:00
|
|
|
C != CEnd; ++C, ++Field) {
|
|
|
|
if (C->capturesThis()) {
|
|
|
|
ThisCapture = *Field;
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
Captures[C->getCapturedVar()] = *Field;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2010-03-15 17:07:48 +08:00
|
|
|
static CanQualType GetConversionType(ASTContext &Context, NamedDecl *Conv) {
|
|
|
|
QualType T;
|
2010-03-31 09:36:47 +08:00
|
|
|
if (isa<UsingShadowDecl>(Conv))
|
|
|
|
Conv = cast<UsingShadowDecl>(Conv)->getTargetDecl();
|
2010-03-15 17:07:48 +08:00
|
|
|
if (FunctionTemplateDecl *ConvTemp = dyn_cast<FunctionTemplateDecl>(Conv))
|
|
|
|
T = ConvTemp->getTemplatedDecl()->getResultType();
|
|
|
|
else
|
|
|
|
T = cast<CXXConversionDecl>(Conv)->getConversionType();
|
|
|
|
return Context.getCanonicalType(T);
|
2009-10-08 04:43:36 +08:00
|
|
|
}
|
|
|
|
|
2010-03-15 17:07:48 +08:00
|
|
|
/// Collect the visible conversions of a base class.
|
|
|
|
///
|
|
|
|
/// \param Base a base class of the class we're considering
|
|
|
|
/// \param InVirtual whether this base class is a virtual base (or a base
|
|
|
|
/// of a virtual base)
|
|
|
|
/// \param Access the access along the inheritance path to this base
|
|
|
|
/// \param ParentHiddenTypes the conversions provided by the inheritors
|
|
|
|
/// of this base
|
|
|
|
/// \param Output the set to which to add conversions from non-virtual bases
|
|
|
|
/// \param VOutput the set to which to add conversions from virtual bases
|
|
|
|
/// \param HiddenVBaseCs the set of conversions which were hidden in a
|
|
|
|
/// virtual base along some inheritance path
|
|
|
|
static void CollectVisibleConversions(ASTContext &Context,
|
|
|
|
CXXRecordDecl *Record,
|
|
|
|
bool InVirtual,
|
|
|
|
AccessSpecifier Access,
|
|
|
|
const llvm::SmallPtrSet<CanQualType, 8> &ParentHiddenTypes,
|
|
|
|
UnresolvedSetImpl &Output,
|
|
|
|
UnresolvedSetImpl &VOutput,
|
|
|
|
llvm::SmallPtrSet<NamedDecl*, 8> &HiddenVBaseCs) {
|
|
|
|
// The set of types which have conversions in this class or its
|
|
|
|
// subclasses. As an optimization, we don't copy the derived set
|
|
|
|
// unless it might change.
|
|
|
|
const llvm::SmallPtrSet<CanQualType, 8> *HiddenTypes = &ParentHiddenTypes;
|
|
|
|
llvm::SmallPtrSet<CanQualType, 8> HiddenTypesBuffer;
|
|
|
|
|
|
|
|
// Collect the direct conversions and figure out which conversions
|
|
|
|
// will be hidden in the subclasses.
|
|
|
|
UnresolvedSetImpl &Cs = *Record->getConversionFunctions();
|
|
|
|
if (!Cs.empty()) {
|
|
|
|
HiddenTypesBuffer = ParentHiddenTypes;
|
|
|
|
HiddenTypes = &HiddenTypesBuffer;
|
|
|
|
|
|
|
|
for (UnresolvedSetIterator I = Cs.begin(), E = Cs.end(); I != E; ++I) {
|
|
|
|
bool Hidden =
|
|
|
|
!HiddenTypesBuffer.insert(GetConversionType(Context, I.getDecl()));
|
|
|
|
|
|
|
|
// If this conversion is hidden and we're in a virtual base,
|
|
|
|
// remember that it's hidden along some inheritance path.
|
|
|
|
if (Hidden && InVirtual)
|
|
|
|
HiddenVBaseCs.insert(cast<NamedDecl>(I.getDecl()->getCanonicalDecl()));
|
|
|
|
|
|
|
|
// If this conversion isn't hidden, add it to the appropriate output.
|
|
|
|
else if (!Hidden) {
|
|
|
|
AccessSpecifier IAccess
|
|
|
|
= CXXRecordDecl::MergeAccess(Access, I.getAccess());
|
|
|
|
|
|
|
|
if (InVirtual)
|
|
|
|
VOutput.addDecl(I.getDecl(), IAccess);
|
2009-09-13 02:26:03 +08:00
|
|
|
else
|
2010-03-15 17:07:48 +08:00
|
|
|
Output.addDecl(I.getDecl(), IAccess);
|
2009-09-12 05:44:33 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2009-10-26 01:03:50 +08:00
|
|
|
|
2010-03-15 17:07:48 +08:00
|
|
|
// Collect information recursively from any base classes.
|
|
|
|
for (CXXRecordDecl::base_class_iterator
|
|
|
|
I = Record->bases_begin(), E = Record->bases_end(); I != E; ++I) {
|
|
|
|
const RecordType *RT = I->getType()->getAs<RecordType>();
|
|
|
|
if (!RT) continue;
|
2009-10-26 01:03:50 +08:00
|
|
|
|
2010-03-15 17:07:48 +08:00
|
|
|
AccessSpecifier BaseAccess
|
|
|
|
= CXXRecordDecl::MergeAccess(Access, I->getAccessSpecifier());
|
|
|
|
bool BaseInVirtual = InVirtual || I->isVirtual();
|
2009-10-26 01:03:50 +08:00
|
|
|
|
2010-03-15 17:07:48 +08:00
|
|
|
CXXRecordDecl *Base = cast<CXXRecordDecl>(RT->getDecl());
|
|
|
|
CollectVisibleConversions(Context, Base, BaseInVirtual, BaseAccess,
|
|
|
|
*HiddenTypes, Output, VOutput, HiddenVBaseCs);
|
2009-09-12 05:44:33 +08:00
|
|
|
}
|
2010-03-15 17:07:48 +08:00
|
|
|
}
|
2009-10-26 01:03:50 +08:00
|
|
|
|
2010-03-15 17:07:48 +08:00
|
|
|
/// Collect the visible conversions of a class.
|
|
|
|
///
|
|
|
|
/// This would be extremely straightforward if it weren't for virtual
|
|
|
|
/// bases. It might be worth special-casing that, really.
|
|
|
|
static void CollectVisibleConversions(ASTContext &Context,
|
|
|
|
CXXRecordDecl *Record,
|
|
|
|
UnresolvedSetImpl &Output) {
|
|
|
|
// The collection of all conversions in virtual bases that we've
|
|
|
|
// found. These will be added to the output as long as they don't
|
|
|
|
// appear in the hidden-conversions set.
|
|
|
|
UnresolvedSet<8> VBaseCs;
|
|
|
|
|
|
|
|
// The set of conversions in virtual bases that we've determined to
|
|
|
|
// be hidden.
|
|
|
|
llvm::SmallPtrSet<NamedDecl*, 8> HiddenVBaseCs;
|
|
|
|
|
|
|
|
// The set of types hidden by classes derived from this one.
|
|
|
|
llvm::SmallPtrSet<CanQualType, 8> HiddenTypes;
|
|
|
|
|
|
|
|
// Go ahead and collect the direct conversions and add them to the
|
|
|
|
// hidden-types set.
|
|
|
|
UnresolvedSetImpl &Cs = *Record->getConversionFunctions();
|
|
|
|
Output.append(Cs.begin(), Cs.end());
|
|
|
|
for (UnresolvedSetIterator I = Cs.begin(), E = Cs.end(); I != E; ++I)
|
|
|
|
HiddenTypes.insert(GetConversionType(Context, I.getDecl()));
|
|
|
|
|
|
|
|
// Recursively collect conversions from base classes.
|
|
|
|
for (CXXRecordDecl::base_class_iterator
|
|
|
|
I = Record->bases_begin(), E = Record->bases_end(); I != E; ++I) {
|
|
|
|
const RecordType *RT = I->getType()->getAs<RecordType>();
|
|
|
|
if (!RT) continue;
|
|
|
|
|
|
|
|
CollectVisibleConversions(Context, cast<CXXRecordDecl>(RT->getDecl()),
|
|
|
|
I->isVirtual(), I->getAccessSpecifier(),
|
|
|
|
HiddenTypes, Output, VBaseCs, HiddenVBaseCs);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Add any unhidden conversions provided by virtual bases.
|
|
|
|
for (UnresolvedSetIterator I = VBaseCs.begin(), E = VBaseCs.end();
|
|
|
|
I != E; ++I) {
|
|
|
|
if (!HiddenVBaseCs.count(cast<NamedDecl>(I.getDecl()->getCanonicalDecl())))
|
|
|
|
Output.addDecl(I.getDecl(), I.getAccess());
|
2009-09-12 05:44:33 +08:00
|
|
|
}
|
2009-09-13 02:26:03 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
/// getVisibleConversionFunctions - get all conversion functions visible
|
|
|
|
/// in current class; including conversion function templates.
|
2010-01-20 08:46:10 +08:00
|
|
|
const UnresolvedSetImpl *CXXRecordDecl::getVisibleConversionFunctions() {
|
2009-09-13 02:26:03 +08:00
|
|
|
// If root class, all conversions are visible.
|
|
|
|
if (bases_begin() == bases_end())
|
2010-02-05 06:26:26 +08:00
|
|
|
return &data().Conversions;
|
2009-09-13 02:26:03 +08:00
|
|
|
// If visible conversion list is already evaluated, return it.
|
2010-02-05 06:26:26 +08:00
|
|
|
if (data().ComputedVisibleConversions)
|
|
|
|
return &data().VisibleConversions;
|
2010-03-15 17:07:48 +08:00
|
|
|
CollectVisibleConversions(getASTContext(), this, data().VisibleConversions);
|
2010-02-05 06:26:26 +08:00
|
|
|
data().ComputedVisibleConversions = true;
|
|
|
|
return &data().VisibleConversions;
|
2009-09-12 05:44:33 +08:00
|
|
|
}
|
|
|
|
|
2010-03-31 09:36:47 +08:00
|
|
|
void CXXRecordDecl::removeConversion(const NamedDecl *ConvDecl) {
|
|
|
|
// This operation is O(N) but extremely rare. Sema only uses it to
|
|
|
|
// remove UsingShadowDecls in a class that were followed by a direct
|
|
|
|
// declaration, e.g.:
|
|
|
|
// class A : B {
|
|
|
|
// using B::operator int;
|
|
|
|
// operator int();
|
|
|
|
// };
|
|
|
|
// This is uncommon by itself and even more uncommon in conjunction
|
|
|
|
// with sufficiently large numbers of directly-declared conversions
|
|
|
|
// that asymptotic behavior matters.
|
|
|
|
|
|
|
|
UnresolvedSetImpl &Convs = *getConversionFunctions();
|
|
|
|
for (unsigned I = 0, E = Convs.size(); I != E; ++I) {
|
|
|
|
if (Convs[I].getDecl() == ConvDecl) {
|
|
|
|
Convs.erase(I);
|
|
|
|
assert(std::find(Convs.begin(), Convs.end(), ConvDecl) == Convs.end()
|
|
|
|
&& "conversion was found multiple times in unresolved set");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
2008-11-08 04:08:42 +08:00
|
|
|
|
2010-03-31 09:36:47 +08:00
|
|
|
llvm_unreachable("conversion not found in set!");
|
2009-08-22 07:19:43 +08:00
|
|
|
}
|
2009-06-20 03:55:27 +08:00
|
|
|
|
2009-10-08 23:14:33 +08:00
|
|
|
CXXRecordDecl *CXXRecordDecl::getInstantiatedFromMemberClass() const {
|
2009-10-13 04:18:28 +08:00
|
|
|
if (MemberSpecializationInfo *MSInfo = getMemberSpecializationInfo())
|
2009-10-08 23:14:33 +08:00
|
|
|
return cast<CXXRecordDecl>(MSInfo->getInstantiatedFrom());
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2009-10-13 04:18:28 +08:00
|
|
|
MemberSpecializationInfo *CXXRecordDecl::getMemberSpecializationInfo() const {
|
|
|
|
return TemplateOrInstantiation.dyn_cast<MemberSpecializationInfo *>();
|
|
|
|
}
|
|
|
|
|
2009-10-08 23:14:33 +08:00
|
|
|
void
|
|
|
|
CXXRecordDecl::setInstantiationOfMemberClass(CXXRecordDecl *RD,
|
|
|
|
TemplateSpecializationKind TSK) {
|
|
|
|
assert(TemplateOrInstantiation.isNull() &&
|
|
|
|
"Previous template or instantiation?");
|
|
|
|
assert(!isa<ClassTemplateSpecializationDecl>(this));
|
|
|
|
TemplateOrInstantiation
|
|
|
|
= new (getASTContext()) MemberSpecializationInfo(RD, TSK);
|
|
|
|
}
|
|
|
|
|
2009-12-07 14:33:48 +08:00
|
|
|
TemplateSpecializationKind CXXRecordDecl::getTemplateSpecializationKind() const{
|
|
|
|
if (const ClassTemplateSpecializationDecl *Spec
|
2009-10-08 23:14:33 +08:00
|
|
|
= dyn_cast<ClassTemplateSpecializationDecl>(this))
|
|
|
|
return Spec->getSpecializationKind();
|
|
|
|
|
2009-10-13 04:18:28 +08:00
|
|
|
if (MemberSpecializationInfo *MSInfo = getMemberSpecializationInfo())
|
2009-10-08 23:14:33 +08:00
|
|
|
return MSInfo->getTemplateSpecializationKind();
|
|
|
|
|
|
|
|
return TSK_Undeclared;
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
CXXRecordDecl::setTemplateSpecializationKind(TemplateSpecializationKind TSK) {
|
|
|
|
if (ClassTemplateSpecializationDecl *Spec
|
|
|
|
= dyn_cast<ClassTemplateSpecializationDecl>(this)) {
|
|
|
|
Spec->setSpecializationKind(TSK);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2009-10-13 04:18:28 +08:00
|
|
|
if (MemberSpecializationInfo *MSInfo = getMemberSpecializationInfo()) {
|
2009-10-08 23:14:33 +08:00
|
|
|
MSInfo->setTemplateSpecializationKind(TSK);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2011-09-23 13:06:16 +08:00
|
|
|
llvm_unreachable("Not a class template or member class specialization");
|
2009-10-08 23:14:33 +08:00
|
|
|
}
|
|
|
|
|
2010-07-01 22:13:13 +08:00
|
|
|
CXXDestructorDecl *CXXRecordDecl::getDestructor() const {
|
|
|
|
ASTContext &Context = getASTContext();
|
2009-05-30 05:03:38 +08:00
|
|
|
QualType ClassType = Context.getTypeDeclType(this);
|
2009-09-09 23:08:12 +08:00
|
|
|
|
|
|
|
DeclarationName Name
|
2009-08-05 13:36:45 +08:00
|
|
|
= Context.DeclarationNames.getCXXDestructorName(
|
|
|
|
Context.getCanonicalType(ClassType));
|
2009-05-30 05:03:38 +08:00
|
|
|
|
2010-02-23 08:48:20 +08:00
|
|
|
DeclContext::lookup_const_iterator I, E;
|
2009-09-09 23:08:12 +08:00
|
|
|
llvm::tie(I, E) = lookup(Name);
|
2010-09-03 07:19:42 +08:00
|
|
|
if (I == E)
|
|
|
|
return 0;
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-12-03 01:15:43 +08:00
|
|
|
CXXDestructorDecl *Dtor = cast<CXXDestructorDecl>(*I);
|
2009-05-30 05:03:38 +08:00
|
|
|
return Dtor;
|
|
|
|
}
|
|
|
|
|
2011-02-20 02:51:44 +08:00
|
|
|
void CXXRecordDecl::completeDefinition() {
|
|
|
|
completeDefinition(0);
|
|
|
|
}
|
|
|
|
|
|
|
|
void CXXRecordDecl::completeDefinition(CXXFinalOverriderMap *FinalOverriders) {
|
|
|
|
RecordDecl::completeDefinition();
|
|
|
|
|
2012-03-11 15:00:24 +08:00
|
|
|
if (hasObjectMember() && getASTContext().getLangOpts().ObjCAutoRefCount) {
|
2011-06-16 07:02:42 +08:00
|
|
|
// Objective-C Automatic Reference Counting:
|
|
|
|
// If a class has a non-static data member of Objective-C pointer
|
|
|
|
// type (or array thereof), it is a non-POD type and its
|
|
|
|
// default constructor (if any), copy constructor, copy assignment
|
|
|
|
// operator, and destructor are non-trivial.
|
|
|
|
struct DefinitionData &Data = data();
|
|
|
|
Data.PlainOldData = false;
|
|
|
|
Data.HasTrivialDefaultConstructor = false;
|
|
|
|
Data.HasTrivialCopyConstructor = false;
|
|
|
|
Data.HasTrivialCopyAssignment = false;
|
|
|
|
Data.HasTrivialDestructor = false;
|
2012-02-25 15:33:38 +08:00
|
|
|
Data.HasIrrelevantDestructor = false;
|
2011-06-16 07:02:42 +08:00
|
|
|
}
|
|
|
|
|
2010-09-29 08:15:42 +08:00
|
|
|
// If the class may be abstract (but hasn't been marked as such), check for
|
|
|
|
// any pure final overriders.
|
|
|
|
if (mayBeAbstract()) {
|
|
|
|
CXXFinalOverriderMap MyFinalOverriders;
|
|
|
|
if (!FinalOverriders) {
|
|
|
|
getFinalOverriders(MyFinalOverriders);
|
|
|
|
FinalOverriders = &MyFinalOverriders;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool Done = false;
|
|
|
|
for (CXXFinalOverriderMap::iterator M = FinalOverriders->begin(),
|
|
|
|
MEnd = FinalOverriders->end();
|
|
|
|
M != MEnd && !Done; ++M) {
|
|
|
|
for (OverridingMethods::iterator SO = M->second.begin(),
|
|
|
|
SOEnd = M->second.end();
|
|
|
|
SO != SOEnd && !Done; ++SO) {
|
|
|
|
assert(SO->second.size() > 0 &&
|
|
|
|
"All virtual functions have overridding virtual functions");
|
|
|
|
|
|
|
|
// C++ [class.abstract]p4:
|
|
|
|
// A class is abstract if it contains or inherits at least one
|
|
|
|
// pure virtual function for which the final overrider is pure
|
|
|
|
// virtual.
|
|
|
|
if (SO->second.front().Method->isPure()) {
|
|
|
|
data().Abstract = true;
|
|
|
|
Done = true;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2010-09-29 12:25:11 +08:00
|
|
|
|
|
|
|
// Set access bits correctly on the directly-declared conversions.
|
|
|
|
for (UnresolvedSetIterator I = data().Conversions.begin(),
|
|
|
|
E = data().Conversions.end();
|
|
|
|
I != E; ++I)
|
|
|
|
data().Conversions.setAccess(I, (*I)->getAccess());
|
2010-09-29 08:15:42 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
bool CXXRecordDecl::mayBeAbstract() const {
|
|
|
|
if (data().Abstract || isInvalidDecl() || !data().Polymorphic ||
|
|
|
|
isDependentContext())
|
|
|
|
return false;
|
|
|
|
|
|
|
|
for (CXXRecordDecl::base_class_const_iterator B = bases_begin(),
|
|
|
|
BEnd = bases_end();
|
|
|
|
B != BEnd; ++B) {
|
|
|
|
CXXRecordDecl *BaseDecl
|
|
|
|
= cast<CXXRecordDecl>(B->getType()->getAs<RecordType>()->getDecl());
|
|
|
|
if (BaseDecl->isAbstract())
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2011-12-20 10:48:34 +08:00
|
|
|
void CXXMethodDecl::anchor() { }
|
|
|
|
|
Change struct forward declarations and definitions to use unique RecordDecls, as opposed to creating a single RecordDecl and reusing it.
This change effects both RecordDecls and CXXRecordDecls, but does not effect EnumDecls (yet).
The motivation of this patch is as follows:
- Capture more source information, necessary for refactoring/rewriting clients.
- Pave the way to resolve ownership issues with RecordDecls with the forthcoming
addition of DeclGroups.
Current caveats:
- Until DeclGroups are in place, we will leak RecordDecls not explicitly
referenced by the AST. For example:
typedef struct { ... } x;
The RecordDecl for the struct will be leaked because the TypedefDecl doesn't
refer to it. This will be solved with DeclGroups.
- This patch also (temporarily) breaks CodeGen. More below.
High-level changes:
- As before, TagType still refers to a TagDecl, but it doesn't own it. When
a struct/union/class is first referenced, a RecordType and RecordDecl are
created for it, and the RecordType refers to that RecordDecl. Later, if
a new RecordDecl is created, the pointer to a RecordDecl in RecordType is
updated to point to the RecordDecl that defines the struct/union/class.
- TagDecl and RecordDecl now how a method 'getDefinition()' to return the
TagDecl*/RecordDecl* that refers to the TagDecl* that defines a particular
enum/struct/class/union. This is useful from going from a RecordDecl* that
defines a forward declaration to the RecordDecl* that provides the actual
definition. Note that this also works for EnumDecls, except that in this case
there is no distinction between forward declarations and definitions (yet).
- Clients should no longer assume that 'isDefinition()' returns true from a
RecordDecl if the corresponding struct/union/class has been defined.
isDefinition() only returns true if a particular RecordDecl is the defining
Decl. Use 'getDefinition()' instead to determine if a struct has been defined.
- The main changes to Sema happen in ActOnTag. To make the changes more
incremental, I split off the processing of enums and structs et al into two
code paths. Enums use the original code path (which is in ActOnTag) and
structs use the ActOnTagStruct. Eventually the two code paths will be merged,
but the idea was to preserve the original logic both for comparison and not to
change the logic for both enums and structs all at once.
- There is NO CHAINING of RecordDecls for the same RecordType. All RecordDecls
that correspond to the same type simply have a pointer to that type. If we
need to figure out what are all the RecordDecls for a given type we can build
a backmap.
- The diff in CXXRecordDecl.[cpp,h] is actually very small; it just mimics the
changes to RecordDecl. For some reason 'svn' marks the entire file as changed.
Why is CodeGen broken:
- Codegen assumes that there is an equivalence between RecordDecl* and
RecordType*. This was true before because we only created one RecordDecl* for
a given RecordType*, but it is no longer true. I believe this shouldn't be too
hard to change, but the patch was big enough as it is.
I have tested this patch on both the clang test suite, and by running the static analyzer over Postgresql and a large Apple-internal project (mix of Objective-C and C).
llvm-svn: 55839
2008-09-06 01:16:31 +08:00
|
|
|
CXXMethodDecl *
|
|
|
|
CXXMethodDecl::Create(ASTContext &C, CXXRecordDecl *RD,
|
2011-03-08 16:55:46 +08:00
|
|
|
SourceLocation StartLoc,
|
2010-08-12 06:01:17 +08:00
|
|
|
const DeclarationNameInfo &NameInfo,
|
2009-12-07 10:54:59 +08:00
|
|
|
QualType T, TypeSourceInfo *TInfo,
|
2011-03-09 01:10:18 +08:00
|
|
|
bool isStatic, StorageClass SCAsWritten, bool isInline,
|
2011-08-16 05:04:07 +08:00
|
|
|
bool isConstexpr, SourceLocation EndLocation) {
|
2011-03-08 16:55:46 +08:00
|
|
|
return new (C) CXXMethodDecl(CXXMethod, RD, StartLoc, NameInfo, T, TInfo,
|
2011-08-16 05:04:07 +08:00
|
|
|
isStatic, SCAsWritten, isInline, isConstexpr,
|
|
|
|
EndLocation);
|
Change struct forward declarations and definitions to use unique RecordDecls, as opposed to creating a single RecordDecl and reusing it.
This change effects both RecordDecls and CXXRecordDecls, but does not effect EnumDecls (yet).
The motivation of this patch is as follows:
- Capture more source information, necessary for refactoring/rewriting clients.
- Pave the way to resolve ownership issues with RecordDecls with the forthcoming
addition of DeclGroups.
Current caveats:
- Until DeclGroups are in place, we will leak RecordDecls not explicitly
referenced by the AST. For example:
typedef struct { ... } x;
The RecordDecl for the struct will be leaked because the TypedefDecl doesn't
refer to it. This will be solved with DeclGroups.
- This patch also (temporarily) breaks CodeGen. More below.
High-level changes:
- As before, TagType still refers to a TagDecl, but it doesn't own it. When
a struct/union/class is first referenced, a RecordType and RecordDecl are
created for it, and the RecordType refers to that RecordDecl. Later, if
a new RecordDecl is created, the pointer to a RecordDecl in RecordType is
updated to point to the RecordDecl that defines the struct/union/class.
- TagDecl and RecordDecl now how a method 'getDefinition()' to return the
TagDecl*/RecordDecl* that refers to the TagDecl* that defines a particular
enum/struct/class/union. This is useful from going from a RecordDecl* that
defines a forward declaration to the RecordDecl* that provides the actual
definition. Note that this also works for EnumDecls, except that in this case
there is no distinction between forward declarations and definitions (yet).
- Clients should no longer assume that 'isDefinition()' returns true from a
RecordDecl if the corresponding struct/union/class has been defined.
isDefinition() only returns true if a particular RecordDecl is the defining
Decl. Use 'getDefinition()' instead to determine if a struct has been defined.
- The main changes to Sema happen in ActOnTag. To make the changes more
incremental, I split off the processing of enums and structs et al into two
code paths. Enums use the original code path (which is in ActOnTag) and
structs use the ActOnTagStruct. Eventually the two code paths will be merged,
but the idea was to preserve the original logic both for comparison and not to
change the logic for both enums and structs all at once.
- There is NO CHAINING of RecordDecls for the same RecordType. All RecordDecls
that correspond to the same type simply have a pointer to that type. If we
need to figure out what are all the RecordDecls for a given type we can build
a backmap.
- The diff in CXXRecordDecl.[cpp,h] is actually very small; it just mimics the
changes to RecordDecl. For some reason 'svn' marks the entire file as changed.
Why is CodeGen broken:
- Codegen assumes that there is an equivalence between RecordDecl* and
RecordType*. This was true before because we only created one RecordDecl* for
a given RecordType*, but it is no longer true. I believe this shouldn't be too
hard to change, but the patch was big enough as it is.
I have tested this patch on both the clang test suite, and by running the static analyzer over Postgresql and a large Apple-internal project (mix of Objective-C and C).
llvm-svn: 55839
2008-09-06 01:16:31 +08:00
|
|
|
}
|
|
|
|
|
2012-01-06 05:55:30 +08:00
|
|
|
CXXMethodDecl *CXXMethodDecl::CreateDeserialized(ASTContext &C, unsigned ID) {
|
|
|
|
void *Mem = AllocateDeserializedDecl(C, ID, sizeof(CXXMethodDecl));
|
|
|
|
return new (Mem) CXXMethodDecl(CXXMethod, 0, SourceLocation(),
|
|
|
|
DeclarationNameInfo(), QualType(),
|
|
|
|
0, false, SC_None, false, false,
|
|
|
|
SourceLocation());
|
|
|
|
}
|
|
|
|
|
2009-09-30 02:16:17 +08:00
|
|
|
bool CXXMethodDecl::isUsualDeallocationFunction() const {
|
|
|
|
if (getOverloadedOperator() != OO_Delete &&
|
|
|
|
getOverloadedOperator() != OO_Array_Delete)
|
|
|
|
return false;
|
2010-02-26 13:06:18 +08:00
|
|
|
|
|
|
|
// C++ [basic.stc.dynamic.deallocation]p2:
|
|
|
|
// A template instance is never a usual deallocation function,
|
|
|
|
// regardless of its signature.
|
|
|
|
if (getPrimaryTemplate())
|
|
|
|
return false;
|
|
|
|
|
2009-09-30 02:16:17 +08:00
|
|
|
// C++ [basic.stc.dynamic.deallocation]p2:
|
|
|
|
// If a class T has a member deallocation function named operator delete
|
|
|
|
// with exactly one parameter, then that function is a usual (non-placement)
|
|
|
|
// deallocation function. [...]
|
|
|
|
if (getNumParams() == 1)
|
|
|
|
return true;
|
|
|
|
|
|
|
|
// C++ [basic.stc.dynamic.deallocation]p2:
|
|
|
|
// [...] If class T does not declare such an operator delete but does
|
|
|
|
// declare a member deallocation function named operator delete with
|
|
|
|
// exactly two parameters, the second of which has type std::size_t (18.1),
|
|
|
|
// then this function is a usual deallocation function.
|
|
|
|
ASTContext &Context = getASTContext();
|
|
|
|
if (getNumParams() != 2 ||
|
2010-02-09 02:54:05 +08:00
|
|
|
!Context.hasSameUnqualifiedType(getParamDecl(1)->getType(),
|
|
|
|
Context.getSizeType()))
|
2009-09-30 02:16:17 +08:00
|
|
|
return false;
|
|
|
|
|
|
|
|
// This function is a usual deallocation function if there are no
|
|
|
|
// single-parameter deallocation functions of the same kind.
|
|
|
|
for (DeclContext::lookup_const_result R = getDeclContext()->lookup(getDeclName());
|
|
|
|
R.first != R.second; ++R.first) {
|
|
|
|
if (const FunctionDecl *FD = dyn_cast<FunctionDecl>(*R.first))
|
|
|
|
if (FD->getNumParams() == 1)
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
Complete reimplementation of the synthesis for implicitly-defined copy
assignment operators.
Previously, Sema provided type-checking and template instantiation for
copy assignment operators, then CodeGen would synthesize the actual
body of the copy constructor. Unfortunately, the two were not in sync,
and CodeGen might pick a copy-assignment operator that is different
from what Sema chose, leading to strange failures, e.g., link-time
failures when CodeGen called a copy-assignment operator that was not
instantiation, run-time failures when copy-assignment operators were
overloaded for const/non-const references and the wrong one was
picked, and run-time failures when by-value copy-assignment operators
did not have their arguments properly copy-initialized.
This implementation synthesizes the implicitly-defined copy assignment
operator bodies in Sema, so that the resulting ASTs encode exactly
what CodeGen needs to do; there is no longer any special code in
CodeGen to synthesize copy-assignment operators. The synthesis of the
body is relatively simple, and we generate one of three different
kinds of copy statements for each base or member:
- For a class subobject, call the appropriate copy-assignment
operator, after overload resolution has determined what that is.
- For an array of scalar types or an array of class types that have
trivial copy assignment operators, construct a call to
__builtin_memcpy.
- For an array of class types with non-trivial copy assignment
operators, synthesize a (possibly nested!) for loop whose inner
statement calls the copy constructor.
- For a scalar type, use built-in assignment.
This patch fixes at least a few tests cases in Boost.Spirit that were
failing because CodeGen picked the wrong copy-assignment operator
(leading to link-time failures), and I suspect a number of undiagnosed
problems will also go away with this change.
Some of the diagnostics we had previously have gotten worse with this
change, since we're going through generic code for our
type-checking. I will improve this in a subsequent patch.
llvm-svn: 102853
2010-05-02 04:49:11 +08:00
|
|
|
bool CXXMethodDecl::isCopyAssignmentOperator() const {
|
2011-05-26 04:50:04 +08:00
|
|
|
// C++0x [class.copy]p17:
|
Complete reimplementation of the synthesis for implicitly-defined copy
assignment operators.
Previously, Sema provided type-checking and template instantiation for
copy assignment operators, then CodeGen would synthesize the actual
body of the copy constructor. Unfortunately, the two were not in sync,
and CodeGen might pick a copy-assignment operator that is different
from what Sema chose, leading to strange failures, e.g., link-time
failures when CodeGen called a copy-assignment operator that was not
instantiation, run-time failures when copy-assignment operators were
overloaded for const/non-const references and the wrong one was
picked, and run-time failures when by-value copy-assignment operators
did not have their arguments properly copy-initialized.
This implementation synthesizes the implicitly-defined copy assignment
operator bodies in Sema, so that the resulting ASTs encode exactly
what CodeGen needs to do; there is no longer any special code in
CodeGen to synthesize copy-assignment operators. The synthesis of the
body is relatively simple, and we generate one of three different
kinds of copy statements for each base or member:
- For a class subobject, call the appropriate copy-assignment
operator, after overload resolution has determined what that is.
- For an array of scalar types or an array of class types that have
trivial copy assignment operators, construct a call to
__builtin_memcpy.
- For an array of class types with non-trivial copy assignment
operators, synthesize a (possibly nested!) for loop whose inner
statement calls the copy constructor.
- For a scalar type, use built-in assignment.
This patch fixes at least a few tests cases in Boost.Spirit that were
failing because CodeGen picked the wrong copy-assignment operator
(leading to link-time failures), and I suspect a number of undiagnosed
problems will also go away with this change.
Some of the diagnostics we had previously have gotten worse with this
change, since we're going through generic code for our
type-checking. I will improve this in a subsequent patch.
llvm-svn: 102853
2010-05-02 04:49:11 +08:00
|
|
|
// A user-declared copy assignment operator X::operator= is a non-static
|
|
|
|
// non-template member function of class X with exactly one parameter of
|
|
|
|
// type X, X&, const X&, volatile X& or const volatile X&.
|
|
|
|
if (/*operator=*/getOverloadedOperator() != OO_Equal ||
|
|
|
|
/*non-static*/ isStatic() ||
|
2011-05-26 04:50:04 +08:00
|
|
|
/*non-template*/getPrimaryTemplate() || getDescribedFunctionTemplate())
|
Complete reimplementation of the synthesis for implicitly-defined copy
assignment operators.
Previously, Sema provided type-checking and template instantiation for
copy assignment operators, then CodeGen would synthesize the actual
body of the copy constructor. Unfortunately, the two were not in sync,
and CodeGen might pick a copy-assignment operator that is different
from what Sema chose, leading to strange failures, e.g., link-time
failures when CodeGen called a copy-assignment operator that was not
instantiation, run-time failures when copy-assignment operators were
overloaded for const/non-const references and the wrong one was
picked, and run-time failures when by-value copy-assignment operators
did not have their arguments properly copy-initialized.
This implementation synthesizes the implicitly-defined copy assignment
operator bodies in Sema, so that the resulting ASTs encode exactly
what CodeGen needs to do; there is no longer any special code in
CodeGen to synthesize copy-assignment operators. The synthesis of the
body is relatively simple, and we generate one of three different
kinds of copy statements for each base or member:
- For a class subobject, call the appropriate copy-assignment
operator, after overload resolution has determined what that is.
- For an array of scalar types or an array of class types that have
trivial copy assignment operators, construct a call to
__builtin_memcpy.
- For an array of class types with non-trivial copy assignment
operators, synthesize a (possibly nested!) for loop whose inner
statement calls the copy constructor.
- For a scalar type, use built-in assignment.
This patch fixes at least a few tests cases in Boost.Spirit that were
failing because CodeGen picked the wrong copy-assignment operator
(leading to link-time failures), and I suspect a number of undiagnosed
problems will also go away with this change.
Some of the diagnostics we had previously have gotten worse with this
change, since we're going through generic code for our
type-checking. I will improve this in a subsequent patch.
llvm-svn: 102853
2010-05-02 04:49:11 +08:00
|
|
|
return false;
|
|
|
|
|
|
|
|
QualType ParamType = getParamDecl(0)->getType();
|
|
|
|
if (const LValueReferenceType *Ref = ParamType->getAs<LValueReferenceType>())
|
|
|
|
ParamType = Ref->getPointeeType();
|
|
|
|
|
|
|
|
ASTContext &Context = getASTContext();
|
|
|
|
QualType ClassType
|
|
|
|
= Context.getCanonicalType(Context.getTypeDeclType(getParent()));
|
|
|
|
return Context.hasSameUnqualifiedType(ClassType, ParamType);
|
|
|
|
}
|
|
|
|
|
2011-05-26 04:50:04 +08:00
|
|
|
bool CXXMethodDecl::isMoveAssignmentOperator() const {
|
|
|
|
// C++0x [class.copy]p19:
|
|
|
|
// A user-declared move assignment operator X::operator= is a non-static
|
|
|
|
// non-template member function of class X with exactly one parameter of type
|
|
|
|
// X&&, const X&&, volatile X&&, or const volatile X&&.
|
|
|
|
if (getOverloadedOperator() != OO_Equal || isStatic() ||
|
|
|
|
getPrimaryTemplate() || getDescribedFunctionTemplate())
|
|
|
|
return false;
|
|
|
|
|
|
|
|
QualType ParamType = getParamDecl(0)->getType();
|
|
|
|
if (!isa<RValueReferenceType>(ParamType))
|
|
|
|
return false;
|
|
|
|
ParamType = ParamType->getPointeeType();
|
|
|
|
|
|
|
|
ASTContext &Context = getASTContext();
|
|
|
|
QualType ClassType
|
|
|
|
= Context.getCanonicalType(Context.getTypeDeclType(getParent()));
|
|
|
|
return Context.hasSameUnqualifiedType(ClassType, ParamType);
|
|
|
|
}
|
|
|
|
|
2009-05-17 07:58:37 +08:00
|
|
|
void CXXMethodDecl::addOverriddenMethod(const CXXMethodDecl *MD) {
|
2009-12-04 13:51:56 +08:00
|
|
|
assert(MD->isCanonicalDecl() && "Method is not canonical!");
|
2010-01-31 01:42:34 +08:00
|
|
|
assert(!MD->getParent()->isDependentContext() &&
|
|
|
|
"Can't add an overridden method to a class template!");
|
2012-03-10 09:39:01 +08:00
|
|
|
assert(MD->isVirtual() && "Method is not virtual!");
|
2010-01-31 01:42:34 +08:00
|
|
|
|
2010-03-03 07:58:15 +08:00
|
|
|
getASTContext().addOverriddenMethod(this, MD);
|
2009-05-17 07:58:37 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
CXXMethodDecl::method_iterator CXXMethodDecl::begin_overridden_methods() const {
|
2012-03-10 09:39:01 +08:00
|
|
|
if (isa<CXXConstructorDecl>(this)) return 0;
|
2010-03-03 07:58:15 +08:00
|
|
|
return getASTContext().overridden_methods_begin(this);
|
2009-05-17 07:58:37 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
CXXMethodDecl::method_iterator CXXMethodDecl::end_overridden_methods() const {
|
2012-03-10 09:39:01 +08:00
|
|
|
if (isa<CXXConstructorDecl>(this)) return 0;
|
2010-03-03 07:58:15 +08:00
|
|
|
return getASTContext().overridden_methods_end(this);
|
2009-05-17 07:58:37 +08:00
|
|
|
}
|
|
|
|
|
2010-07-05 05:44:35 +08:00
|
|
|
unsigned CXXMethodDecl::size_overridden_methods() const {
|
2012-03-10 09:39:01 +08:00
|
|
|
if (isa<CXXConstructorDecl>(this)) return 0;
|
2010-07-05 05:44:35 +08:00
|
|
|
return getASTContext().overridden_methods_size(this);
|
|
|
|
}
|
|
|
|
|
Change struct forward declarations and definitions to use unique RecordDecls, as opposed to creating a single RecordDecl and reusing it.
This change effects both RecordDecls and CXXRecordDecls, but does not effect EnumDecls (yet).
The motivation of this patch is as follows:
- Capture more source information, necessary for refactoring/rewriting clients.
- Pave the way to resolve ownership issues with RecordDecls with the forthcoming
addition of DeclGroups.
Current caveats:
- Until DeclGroups are in place, we will leak RecordDecls not explicitly
referenced by the AST. For example:
typedef struct { ... } x;
The RecordDecl for the struct will be leaked because the TypedefDecl doesn't
refer to it. This will be solved with DeclGroups.
- This patch also (temporarily) breaks CodeGen. More below.
High-level changes:
- As before, TagType still refers to a TagDecl, but it doesn't own it. When
a struct/union/class is first referenced, a RecordType and RecordDecl are
created for it, and the RecordType refers to that RecordDecl. Later, if
a new RecordDecl is created, the pointer to a RecordDecl in RecordType is
updated to point to the RecordDecl that defines the struct/union/class.
- TagDecl and RecordDecl now how a method 'getDefinition()' to return the
TagDecl*/RecordDecl* that refers to the TagDecl* that defines a particular
enum/struct/class/union. This is useful from going from a RecordDecl* that
defines a forward declaration to the RecordDecl* that provides the actual
definition. Note that this also works for EnumDecls, except that in this case
there is no distinction between forward declarations and definitions (yet).
- Clients should no longer assume that 'isDefinition()' returns true from a
RecordDecl if the corresponding struct/union/class has been defined.
isDefinition() only returns true if a particular RecordDecl is the defining
Decl. Use 'getDefinition()' instead to determine if a struct has been defined.
- The main changes to Sema happen in ActOnTag. To make the changes more
incremental, I split off the processing of enums and structs et al into two
code paths. Enums use the original code path (which is in ActOnTag) and
structs use the ActOnTagStruct. Eventually the two code paths will be merged,
but the idea was to preserve the original logic both for comparison and not to
change the logic for both enums and structs all at once.
- There is NO CHAINING of RecordDecls for the same RecordType. All RecordDecls
that correspond to the same type simply have a pointer to that type. If we
need to figure out what are all the RecordDecls for a given type we can build
a backmap.
- The diff in CXXRecordDecl.[cpp,h] is actually very small; it just mimics the
changes to RecordDecl. For some reason 'svn' marks the entire file as changed.
Why is CodeGen broken:
- Codegen assumes that there is an equivalence between RecordDecl* and
RecordType*. This was true before because we only created one RecordDecl* for
a given RecordType*, but it is no longer true. I believe this shouldn't be too
hard to change, but the patch was big enough as it is.
I have tested this patch on both the clang test suite, and by running the static analyzer over Postgresql and a large Apple-internal project (mix of Objective-C and C).
llvm-svn: 55839
2008-09-06 01:16:31 +08:00
|
|
|
QualType CXXMethodDecl::getThisType(ASTContext &C) const {
|
2008-10-25 06:28:18 +08:00
|
|
|
// C++ 9.3.2p1: The type of this in a member function of a class X is X*.
|
|
|
|
// If the member function is declared const, the type of this is const X*,
|
|
|
|
// if the member function is declared volatile, the type of this is
|
|
|
|
// volatile X*, and if the member function is declared const volatile,
|
|
|
|
// the type of this is const volatile X*.
|
|
|
|
|
Change struct forward declarations and definitions to use unique RecordDecls, as opposed to creating a single RecordDecl and reusing it.
This change effects both RecordDecls and CXXRecordDecls, but does not effect EnumDecls (yet).
The motivation of this patch is as follows:
- Capture more source information, necessary for refactoring/rewriting clients.
- Pave the way to resolve ownership issues with RecordDecls with the forthcoming
addition of DeclGroups.
Current caveats:
- Until DeclGroups are in place, we will leak RecordDecls not explicitly
referenced by the AST. For example:
typedef struct { ... } x;
The RecordDecl for the struct will be leaked because the TypedefDecl doesn't
refer to it. This will be solved with DeclGroups.
- This patch also (temporarily) breaks CodeGen. More below.
High-level changes:
- As before, TagType still refers to a TagDecl, but it doesn't own it. When
a struct/union/class is first referenced, a RecordType and RecordDecl are
created for it, and the RecordType refers to that RecordDecl. Later, if
a new RecordDecl is created, the pointer to a RecordDecl in RecordType is
updated to point to the RecordDecl that defines the struct/union/class.
- TagDecl and RecordDecl now how a method 'getDefinition()' to return the
TagDecl*/RecordDecl* that refers to the TagDecl* that defines a particular
enum/struct/class/union. This is useful from going from a RecordDecl* that
defines a forward declaration to the RecordDecl* that provides the actual
definition. Note that this also works for EnumDecls, except that in this case
there is no distinction between forward declarations and definitions (yet).
- Clients should no longer assume that 'isDefinition()' returns true from a
RecordDecl if the corresponding struct/union/class has been defined.
isDefinition() only returns true if a particular RecordDecl is the defining
Decl. Use 'getDefinition()' instead to determine if a struct has been defined.
- The main changes to Sema happen in ActOnTag. To make the changes more
incremental, I split off the processing of enums and structs et al into two
code paths. Enums use the original code path (which is in ActOnTag) and
structs use the ActOnTagStruct. Eventually the two code paths will be merged,
but the idea was to preserve the original logic both for comparison and not to
change the logic for both enums and structs all at once.
- There is NO CHAINING of RecordDecls for the same RecordType. All RecordDecls
that correspond to the same type simply have a pointer to that type. If we
need to figure out what are all the RecordDecls for a given type we can build
a backmap.
- The diff in CXXRecordDecl.[cpp,h] is actually very small; it just mimics the
changes to RecordDecl. For some reason 'svn' marks the entire file as changed.
Why is CodeGen broken:
- Codegen assumes that there is an equivalence between RecordDecl* and
RecordType*. This was true before because we only created one RecordDecl* for
a given RecordType*, but it is no longer true. I believe this shouldn't be too
hard to change, but the patch was big enough as it is.
I have tested this patch on both the clang test suite, and by running the static analyzer over Postgresql and a large Apple-internal project (mix of Objective-C and C).
llvm-svn: 55839
2008-09-06 01:16:31 +08:00
|
|
|
assert(isInstance() && "No 'this' for static methods!");
|
2009-06-13 10:59:33 +08:00
|
|
|
|
2010-03-10 11:28:59 +08:00
|
|
|
QualType ClassTy = C.getTypeDeclType(getParent());
|
2009-09-25 03:53:00 +08:00
|
|
|
ClassTy = C.getQualifiedType(ClassTy,
|
|
|
|
Qualifiers::fromCVRMask(getTypeQualifiers()));
|
2009-07-11 05:35:09 +08:00
|
|
|
return C.getPointerType(ClassTy);
|
Change struct forward declarations and definitions to use unique RecordDecls, as opposed to creating a single RecordDecl and reusing it.
This change effects both RecordDecls and CXXRecordDecls, but does not effect EnumDecls (yet).
The motivation of this patch is as follows:
- Capture more source information, necessary for refactoring/rewriting clients.
- Pave the way to resolve ownership issues with RecordDecls with the forthcoming
addition of DeclGroups.
Current caveats:
- Until DeclGroups are in place, we will leak RecordDecls not explicitly
referenced by the AST. For example:
typedef struct { ... } x;
The RecordDecl for the struct will be leaked because the TypedefDecl doesn't
refer to it. This will be solved with DeclGroups.
- This patch also (temporarily) breaks CodeGen. More below.
High-level changes:
- As before, TagType still refers to a TagDecl, but it doesn't own it. When
a struct/union/class is first referenced, a RecordType and RecordDecl are
created for it, and the RecordType refers to that RecordDecl. Later, if
a new RecordDecl is created, the pointer to a RecordDecl in RecordType is
updated to point to the RecordDecl that defines the struct/union/class.
- TagDecl and RecordDecl now how a method 'getDefinition()' to return the
TagDecl*/RecordDecl* that refers to the TagDecl* that defines a particular
enum/struct/class/union. This is useful from going from a RecordDecl* that
defines a forward declaration to the RecordDecl* that provides the actual
definition. Note that this also works for EnumDecls, except that in this case
there is no distinction between forward declarations and definitions (yet).
- Clients should no longer assume that 'isDefinition()' returns true from a
RecordDecl if the corresponding struct/union/class has been defined.
isDefinition() only returns true if a particular RecordDecl is the defining
Decl. Use 'getDefinition()' instead to determine if a struct has been defined.
- The main changes to Sema happen in ActOnTag. To make the changes more
incremental, I split off the processing of enums and structs et al into two
code paths. Enums use the original code path (which is in ActOnTag) and
structs use the ActOnTagStruct. Eventually the two code paths will be merged,
but the idea was to preserve the original logic both for comparison and not to
change the logic for both enums and structs all at once.
- There is NO CHAINING of RecordDecls for the same RecordType. All RecordDecls
that correspond to the same type simply have a pointer to that type. If we
need to figure out what are all the RecordDecls for a given type we can build
a backmap.
- The diff in CXXRecordDecl.[cpp,h] is actually very small; it just mimics the
changes to RecordDecl. For some reason 'svn' marks the entire file as changed.
Why is CodeGen broken:
- Codegen assumes that there is an equivalence between RecordDecl* and
RecordType*. This was true before because we only created one RecordDecl* for
a given RecordType*, but it is no longer true. I believe this shouldn't be too
hard to change, but the patch was big enough as it is.
I have tested this patch on both the clang test suite, and by running the static analyzer over Postgresql and a large Apple-internal project (mix of Objective-C and C).
llvm-svn: 55839
2008-09-06 01:16:31 +08:00
|
|
|
}
|
|
|
|
|
2009-12-07 04:50:05 +08:00
|
|
|
bool CXXMethodDecl::hasInlineBody() const {
|
2010-01-06 03:06:31 +08:00
|
|
|
// If this function is a template instantiation, look at the template from
|
|
|
|
// which it was instantiated.
|
|
|
|
const FunctionDecl *CheckFn = getTemplateInstantiationPattern();
|
|
|
|
if (!CheckFn)
|
|
|
|
CheckFn = this;
|
|
|
|
|
2009-12-07 04:50:05 +08:00
|
|
|
const FunctionDecl *fn;
|
2010-07-07 19:31:19 +08:00
|
|
|
return CheckFn->hasBody(fn) && !fn->isOutOfLine();
|
2009-12-07 04:50:05 +08:00
|
|
|
}
|
|
|
|
|
2012-02-17 11:02:34 +08:00
|
|
|
bool CXXMethodDecl::isLambdaStaticInvoker() const {
|
|
|
|
return getParent()->isLambda() &&
|
|
|
|
getIdentifier() && getIdentifier()->getName() == "__invoke";
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2011-01-09 04:30:50 +08:00
|
|
|
CXXCtorInitializer::CXXCtorInitializer(ASTContext &Context,
|
|
|
|
TypeSourceInfo *TInfo, bool IsVirtual,
|
|
|
|
SourceLocation L, Expr *Init,
|
|
|
|
SourceLocation R,
|
|
|
|
SourceLocation EllipsisLoc)
|
2011-01-09 07:01:16 +08:00
|
|
|
: Initializee(TInfo), MemberOrEllipsisLocation(EllipsisLoc), Init(Init),
|
2011-11-01 09:16:03 +08:00
|
|
|
LParenLoc(L), RParenLoc(R), IsDelegating(false), IsVirtual(IsVirtual),
|
|
|
|
IsWritten(false), SourceOrderOrNumArrayIndices(0)
|
2009-12-03 06:36:29 +08:00
|
|
|
{
|
2008-11-05 12:29:56 +08:00
|
|
|
}
|
|
|
|
|
2011-01-09 04:30:50 +08:00
|
|
|
CXXCtorInitializer::CXXCtorInitializer(ASTContext &Context,
|
|
|
|
FieldDecl *Member,
|
|
|
|
SourceLocation MemberLoc,
|
|
|
|
SourceLocation L, Expr *Init,
|
|
|
|
SourceLocation R)
|
2011-01-09 07:01:16 +08:00
|
|
|
: Initializee(Member), MemberOrEllipsisLocation(MemberLoc), Init(Init),
|
2011-11-01 09:16:03 +08:00
|
|
|
LParenLoc(L), RParenLoc(R), IsDelegating(false), IsVirtual(false),
|
2010-12-04 17:14:42 +08:00
|
|
|
IsWritten(false), SourceOrderOrNumArrayIndices(0)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
2011-01-09 04:30:50 +08:00
|
|
|
CXXCtorInitializer::CXXCtorInitializer(ASTContext &Context,
|
|
|
|
IndirectFieldDecl *Member,
|
|
|
|
SourceLocation MemberLoc,
|
|
|
|
SourceLocation L, Expr *Init,
|
|
|
|
SourceLocation R)
|
2011-01-09 07:01:16 +08:00
|
|
|
: Initializee(Member), MemberOrEllipsisLocation(MemberLoc), Init(Init),
|
2011-11-01 09:16:03 +08:00
|
|
|
LParenLoc(L), RParenLoc(R), IsDelegating(false), IsVirtual(false),
|
2010-05-27 02:09:23 +08:00
|
|
|
IsWritten(false), SourceOrderOrNumArrayIndices(0)
|
2009-12-03 06:36:29 +08:00
|
|
|
{
|
2008-11-05 12:29:56 +08:00
|
|
|
}
|
|
|
|
|
2011-02-27 03:13:13 +08:00
|
|
|
CXXCtorInitializer::CXXCtorInitializer(ASTContext &Context,
|
2011-11-01 09:16:03 +08:00
|
|
|
TypeSourceInfo *TInfo,
|
|
|
|
SourceLocation L, Expr *Init,
|
2011-02-27 03:13:13 +08:00
|
|
|
SourceLocation R)
|
2011-11-01 09:16:03 +08:00
|
|
|
: Initializee(TInfo), MemberOrEllipsisLocation(), Init(Init),
|
|
|
|
LParenLoc(L), RParenLoc(R), IsDelegating(true), IsVirtual(false),
|
2011-02-27 03:13:13 +08:00
|
|
|
IsWritten(false), SourceOrderOrNumArrayIndices(0)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
2011-01-09 04:30:50 +08:00
|
|
|
CXXCtorInitializer::CXXCtorInitializer(ASTContext &Context,
|
|
|
|
FieldDecl *Member,
|
|
|
|
SourceLocation MemberLoc,
|
|
|
|
SourceLocation L, Expr *Init,
|
|
|
|
SourceLocation R,
|
|
|
|
VarDecl **Indices,
|
|
|
|
unsigned NumIndices)
|
2011-01-09 07:01:16 +08:00
|
|
|
: Initializee(Member), MemberOrEllipsisLocation(MemberLoc), Init(Init),
|
2010-12-04 17:14:42 +08:00
|
|
|
LParenLoc(L), RParenLoc(R), IsVirtual(false),
|
2010-05-27 02:09:23 +08:00
|
|
|
IsWritten(false), SourceOrderOrNumArrayIndices(NumIndices)
|
Reimplement code generation for copying fields in the
implicitly-generated copy constructor. Previously, Sema would perform
some checking and instantiation to determine which copy constructors,
etc., would be called, then CodeGen would attempt to figure out which
copy constructor to call... but would get it wrong, or poke at an
uninstantiated default argument, or fail in other ways.
The new scheme is similar to what we now do for the implicit
copy-assignment operator, where Sema performs all of the semantic
analysis and builds specific ASTs that look similar to the ASTs we'd
get from explicitly writing the copy constructor, so that CodeGen need
only do a direct translation.
However, it's not quite that simple because one cannot explicit write
elementwise copy-construction of an array. So, I've extended
CXXBaseOrMemberInitializer to contain a list of indexing variables
used to copy-construct the elements. For example, if we have:
struct A { A(const A&); };
struct B {
A array[2][3];
};
then we generate an implicit copy assignment operator for B that looks
something like this:
B::B(const B &other) : array[i0][i1](other.array[i0][i1]) { }
CodeGen will loop over the invented variables i0 and i1 to visit all
elements in the array, so that each element in the destination array
will be copy-constructed from the corresponding element in the source
array. Of course, if we're dealing with arrays of scalars or class
types with trivial copy-assignment operators, we just generate a
memcpy rather than a loop.
Fixes PR6928, PR5989, and PR6887. Boost.Regex now compiles and passes
all of its regression tests.
Conspicuously missing from this patch is handling for the exceptional
case, where we need to destruct those objects that we have
constructed. I'll address that case separately.
llvm-svn: 103079
2010-05-05 13:51:00 +08:00
|
|
|
{
|
|
|
|
VarDecl **MyIndices = reinterpret_cast<VarDecl **> (this + 1);
|
|
|
|
memcpy(MyIndices, Indices, NumIndices * sizeof(VarDecl *));
|
|
|
|
}
|
|
|
|
|
2011-01-09 04:30:50 +08:00
|
|
|
CXXCtorInitializer *CXXCtorInitializer::Create(ASTContext &Context,
|
|
|
|
FieldDecl *Member,
|
|
|
|
SourceLocation MemberLoc,
|
|
|
|
SourceLocation L, Expr *Init,
|
|
|
|
SourceLocation R,
|
|
|
|
VarDecl **Indices,
|
|
|
|
unsigned NumIndices) {
|
|
|
|
void *Mem = Context.Allocate(sizeof(CXXCtorInitializer) +
|
Reimplement code generation for copying fields in the
implicitly-generated copy constructor. Previously, Sema would perform
some checking and instantiation to determine which copy constructors,
etc., would be called, then CodeGen would attempt to figure out which
copy constructor to call... but would get it wrong, or poke at an
uninstantiated default argument, or fail in other ways.
The new scheme is similar to what we now do for the implicit
copy-assignment operator, where Sema performs all of the semantic
analysis and builds specific ASTs that look similar to the ASTs we'd
get from explicitly writing the copy constructor, so that CodeGen need
only do a direct translation.
However, it's not quite that simple because one cannot explicit write
elementwise copy-construction of an array. So, I've extended
CXXBaseOrMemberInitializer to contain a list of indexing variables
used to copy-construct the elements. For example, if we have:
struct A { A(const A&); };
struct B {
A array[2][3];
};
then we generate an implicit copy assignment operator for B that looks
something like this:
B::B(const B &other) : array[i0][i1](other.array[i0][i1]) { }
CodeGen will loop over the invented variables i0 and i1 to visit all
elements in the array, so that each element in the destination array
will be copy-constructed from the corresponding element in the source
array. Of course, if we're dealing with arrays of scalars or class
types with trivial copy-assignment operators, we just generate a
memcpy rather than a loop.
Fixes PR6928, PR5989, and PR6887. Boost.Regex now compiles and passes
all of its regression tests.
Conspicuously missing from this patch is handling for the exceptional
case, where we need to destruct those objects that we have
constructed. I'll address that case separately.
llvm-svn: 103079
2010-05-05 13:51:00 +08:00
|
|
|
sizeof(VarDecl *) * NumIndices,
|
2011-01-09 04:30:50 +08:00
|
|
|
llvm::alignOf<CXXCtorInitializer>());
|
2011-01-09 07:01:16 +08:00
|
|
|
return new (Mem) CXXCtorInitializer(Context, Member, MemberLoc, L, Init, R,
|
|
|
|
Indices, NumIndices);
|
Reimplement code generation for copying fields in the
implicitly-generated copy constructor. Previously, Sema would perform
some checking and instantiation to determine which copy constructors,
etc., would be called, then CodeGen would attempt to figure out which
copy constructor to call... but would get it wrong, or poke at an
uninstantiated default argument, or fail in other ways.
The new scheme is similar to what we now do for the implicit
copy-assignment operator, where Sema performs all of the semantic
analysis and builds specific ASTs that look similar to the ASTs we'd
get from explicitly writing the copy constructor, so that CodeGen need
only do a direct translation.
However, it's not quite that simple because one cannot explicit write
elementwise copy-construction of an array. So, I've extended
CXXBaseOrMemberInitializer to contain a list of indexing variables
used to copy-construct the elements. For example, if we have:
struct A { A(const A&); };
struct B {
A array[2][3];
};
then we generate an implicit copy assignment operator for B that looks
something like this:
B::B(const B &other) : array[i0][i1](other.array[i0][i1]) { }
CodeGen will loop over the invented variables i0 and i1 to visit all
elements in the array, so that each element in the destination array
will be copy-constructed from the corresponding element in the source
array. Of course, if we're dealing with arrays of scalars or class
types with trivial copy-assignment operators, we just generate a
memcpy rather than a loop.
Fixes PR6928, PR5989, and PR6887. Boost.Regex now compiles and passes
all of its regression tests.
Conspicuously missing from this patch is handling for the exceptional
case, where we need to destruct those objects that we have
constructed. I'll address that case separately.
llvm-svn: 103079
2010-05-05 13:51:00 +08:00
|
|
|
}
|
|
|
|
|
2011-01-09 04:30:50 +08:00
|
|
|
TypeLoc CXXCtorInitializer::getBaseClassLoc() const {
|
2009-12-03 06:36:29 +08:00
|
|
|
if (isBaseInitializer())
|
2011-01-09 07:01:16 +08:00
|
|
|
return Initializee.get<TypeSourceInfo*>()->getTypeLoc();
|
2009-12-03 06:36:29 +08:00
|
|
|
else
|
|
|
|
return TypeLoc();
|
|
|
|
}
|
|
|
|
|
2011-01-09 04:30:50 +08:00
|
|
|
const Type *CXXCtorInitializer::getBaseClass() const {
|
2009-12-03 06:36:29 +08:00
|
|
|
if (isBaseInitializer())
|
2011-01-09 07:01:16 +08:00
|
|
|
return Initializee.get<TypeSourceInfo*>()->getType().getTypePtr();
|
2009-12-03 06:36:29 +08:00
|
|
|
else
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2011-01-09 04:30:50 +08:00
|
|
|
SourceLocation CXXCtorInitializer::getSourceLocation() const {
|
2011-11-01 09:16:03 +08:00
|
|
|
if (isAnyMemberInitializer())
|
2009-12-03 06:36:29 +08:00
|
|
|
return getMemberLocation();
|
2011-06-12 01:19:42 +08:00
|
|
|
|
|
|
|
if (isInClassMemberInitializer())
|
|
|
|
return getAnyMember()->getLocation();
|
2009-12-03 06:36:29 +08:00
|
|
|
|
2011-11-01 09:16:03 +08:00
|
|
|
if (TypeSourceInfo *TSInfo = Initializee.get<TypeSourceInfo*>())
|
|
|
|
return TSInfo->getTypeLoc().getLocalSourceRange().getBegin();
|
|
|
|
|
|
|
|
return SourceLocation();
|
2009-12-03 06:36:29 +08:00
|
|
|
}
|
|
|
|
|
2011-01-09 04:30:50 +08:00
|
|
|
SourceRange CXXCtorInitializer::getSourceRange() const {
|
2011-06-12 01:19:42 +08:00
|
|
|
if (isInClassMemberInitializer()) {
|
|
|
|
FieldDecl *D = getAnyMember();
|
|
|
|
if (Expr *I = D->getInClassInitializer())
|
|
|
|
return I->getSourceRange();
|
|
|
|
return SourceRange();
|
|
|
|
}
|
|
|
|
|
2009-12-03 06:36:29 +08:00
|
|
|
return SourceRange(getSourceLocation(), getRParenLoc());
|
2008-11-05 12:29:56 +08:00
|
|
|
}
|
|
|
|
|
2011-12-20 10:48:34 +08:00
|
|
|
void CXXConstructorDecl::anchor() { }
|
|
|
|
|
2010-05-08 05:43:38 +08:00
|
|
|
CXXConstructorDecl *
|
2012-01-06 05:55:30 +08:00
|
|
|
CXXConstructorDecl::CreateDeserialized(ASTContext &C, unsigned ID) {
|
|
|
|
void *Mem = AllocateDeserializedDecl(C, ID, sizeof(CXXConstructorDecl));
|
|
|
|
return new (Mem) CXXConstructorDecl(0, SourceLocation(),DeclarationNameInfo(),
|
|
|
|
QualType(), 0, false, false, false,false);
|
2010-05-08 05:43:38 +08:00
|
|
|
}
|
|
|
|
|
2008-10-31 17:07:45 +08:00
|
|
|
CXXConstructorDecl *
|
|
|
|
CXXConstructorDecl::Create(ASTContext &C, CXXRecordDecl *RD,
|
2011-03-08 16:55:46 +08:00
|
|
|
SourceLocation StartLoc,
|
2010-08-12 06:01:17 +08:00
|
|
|
const DeclarationNameInfo &NameInfo,
|
2009-12-07 10:54:59 +08:00
|
|
|
QualType T, TypeSourceInfo *TInfo,
|
2011-08-16 05:04:07 +08:00
|
|
|
bool isExplicit, bool isInline,
|
|
|
|
bool isImplicitlyDeclared, bool isConstexpr) {
|
2010-08-12 06:01:17 +08:00
|
|
|
assert(NameInfo.getName().getNameKind()
|
|
|
|
== DeclarationName::CXXConstructorName &&
|
2008-11-17 22:58:09 +08:00
|
|
|
"Name must refer to a constructor");
|
2011-03-08 16:55:46 +08:00
|
|
|
return new (C) CXXConstructorDecl(RD, StartLoc, NameInfo, T, TInfo,
|
2011-08-16 05:04:07 +08:00
|
|
|
isExplicit, isInline, isImplicitlyDeclared,
|
|
|
|
isConstexpr);
|
2008-10-31 17:07:45 +08:00
|
|
|
}
|
|
|
|
|
2011-11-01 09:16:03 +08:00
|
|
|
CXXConstructorDecl *CXXConstructorDecl::getTargetConstructor() const {
|
|
|
|
assert(isDelegatingConstructor() && "Not a delegating constructor!");
|
|
|
|
Expr *E = (*init_begin())->getInit()->IgnoreImplicit();
|
|
|
|
if (CXXConstructExpr *Construct = dyn_cast<CXXConstructExpr>(E))
|
|
|
|
return Construct->getConstructor();
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2008-11-01 04:25:05 +08:00
|
|
|
bool CXXConstructorDecl::isDefaultConstructor() const {
|
|
|
|
// C++ [class.ctor]p5:
|
2008-11-06 00:20:31 +08:00
|
|
|
// A default constructor for a class X is a constructor of class
|
|
|
|
// X that can be called without an argument.
|
2008-11-01 04:25:05 +08:00
|
|
|
return (getNumParams() == 0) ||
|
2009-08-25 13:12:04 +08:00
|
|
|
(getNumParams() > 0 && getParamDecl(0)->hasDefaultArg());
|
2008-11-01 04:25:05 +08:00
|
|
|
}
|
|
|
|
|
2009-09-09 23:08:12 +08:00
|
|
|
bool
|
2009-12-22 08:34:07 +08:00
|
|
|
CXXConstructorDecl::isCopyConstructor(unsigned &TypeQuals) const {
|
2011-01-22 03:38:21 +08:00
|
|
|
return isCopyOrMoveConstructor(TypeQuals) &&
|
|
|
|
getParamDecl(0)->getType()->isLValueReferenceType();
|
|
|
|
}
|
|
|
|
|
|
|
|
bool CXXConstructorDecl::isMoveConstructor(unsigned &TypeQuals) const {
|
|
|
|
return isCopyOrMoveConstructor(TypeQuals) &&
|
|
|
|
getParamDecl(0)->getType()->isRValueReferenceType();
|
|
|
|
}
|
|
|
|
|
|
|
|
/// \brief Determine whether this is a copy or move constructor.
|
|
|
|
bool CXXConstructorDecl::isCopyOrMoveConstructor(unsigned &TypeQuals) const {
|
2008-11-01 04:25:05 +08:00
|
|
|
// C++ [class.copy]p2:
|
2008-11-06 00:20:31 +08:00
|
|
|
// A non-template constructor for class X is a copy constructor
|
|
|
|
// if its first parameter is of type X&, const X&, volatile X& or
|
|
|
|
// const volatile X&, and either there are no other parameters
|
|
|
|
// or else all other parameters have default arguments (8.3.6).
|
2011-01-22 03:38:21 +08:00
|
|
|
// C++0x [class.copy]p3:
|
|
|
|
// A non-template constructor for class X is a move constructor if its
|
|
|
|
// first parameter is of type X&&, const X&&, volatile X&&, or
|
|
|
|
// const volatile X&&, and either there are no other parameters or else
|
|
|
|
// all other parameters have default arguments.
|
2008-11-01 04:25:05 +08:00
|
|
|
if ((getNumParams() < 1) ||
|
2009-10-14 07:45:19 +08:00
|
|
|
(getNumParams() > 1 && !getParamDecl(1)->hasDefaultArg()) ||
|
2009-11-14 07:59:09 +08:00
|
|
|
(getPrimaryTemplate() != 0) ||
|
2009-10-14 07:45:19 +08:00
|
|
|
(getDescribedFunctionTemplate() != 0))
|
2008-11-01 04:25:05 +08:00
|
|
|
return false;
|
2011-01-22 03:38:21 +08:00
|
|
|
|
2008-11-01 04:25:05 +08:00
|
|
|
const ParmVarDecl *Param = getParamDecl(0);
|
2011-01-22 03:38:21 +08:00
|
|
|
|
|
|
|
// Do we have a reference type?
|
|
|
|
const ReferenceType *ParamRefType = Param->getType()->getAs<ReferenceType>();
|
2009-11-14 07:59:09 +08:00
|
|
|
if (!ParamRefType)
|
|
|
|
return false;
|
2011-01-22 03:38:21 +08:00
|
|
|
|
2009-11-14 07:59:09 +08:00
|
|
|
// Is it a reference to our class type?
|
2009-12-22 08:34:07 +08:00
|
|
|
ASTContext &Context = getASTContext();
|
|
|
|
|
2009-11-14 07:59:09 +08:00
|
|
|
CanQualType PointeeType
|
|
|
|
= Context.getCanonicalType(ParamRefType->getPointeeType());
|
2009-09-16 04:50:23 +08:00
|
|
|
CanQualType ClassTy
|
|
|
|
= Context.getCanonicalType(Context.getTagDeclType(getParent()));
|
2008-11-01 04:25:05 +08:00
|
|
|
if (PointeeType.getUnqualifiedType() != ClassTy)
|
|
|
|
return false;
|
2011-01-22 03:38:21 +08:00
|
|
|
|
2009-09-25 03:53:00 +08:00
|
|
|
// FIXME: other qualifiers?
|
2011-01-22 03:38:21 +08:00
|
|
|
|
|
|
|
// We have a copy or move constructor.
|
2008-11-01 04:25:05 +08:00
|
|
|
TypeQuals = PointeeType.getCVRQualifiers();
|
2011-01-22 03:38:21 +08:00
|
|
|
return true;
|
2008-11-01 04:25:05 +08:00
|
|
|
}
|
|
|
|
|
2009-08-29 00:57:08 +08:00
|
|
|
bool CXXConstructorDecl::isConvertingConstructor(bool AllowExplicit) const {
|
2008-11-01 00:23:19 +08:00
|
|
|
// C++ [class.conv.ctor]p1:
|
|
|
|
// A constructor declared without the function-specifier explicit
|
|
|
|
// that can be called with a single parameter specifies a
|
|
|
|
// conversion from the type of its first parameter to the type of
|
|
|
|
// its class. Such a constructor is called a converting
|
|
|
|
// constructor.
|
2009-08-29 00:57:08 +08:00
|
|
|
if (isExplicit() && !AllowExplicit)
|
2008-11-01 00:23:19 +08:00
|
|
|
return false;
|
|
|
|
|
2009-09-09 23:08:12 +08:00
|
|
|
return (getNumParams() == 0 &&
|
2009-09-22 07:43:11 +08:00
|
|
|
getType()->getAs<FunctionProtoType>()->isVariadic()) ||
|
2008-11-01 00:23:19 +08:00
|
|
|
(getNumParams() == 1) ||
|
2009-06-06 12:14:07 +08:00
|
|
|
(getNumParams() > 1 && getParamDecl(1)->hasDefaultArg());
|
2008-11-01 00:23:19 +08:00
|
|
|
}
|
2008-10-31 17:07:45 +08:00
|
|
|
|
2010-11-09 01:16:59 +08:00
|
|
|
bool CXXConstructorDecl::isSpecializationCopyingObject() const {
|
2009-11-14 09:20:54 +08:00
|
|
|
if ((getNumParams() < 1) ||
|
|
|
|
(getNumParams() > 1 && !getParamDecl(1)->hasDefaultArg()) ||
|
|
|
|
(getPrimaryTemplate() == 0) ||
|
|
|
|
(getDescribedFunctionTemplate() != 0))
|
|
|
|
return false;
|
|
|
|
|
|
|
|
const ParmVarDecl *Param = getParamDecl(0);
|
|
|
|
|
|
|
|
ASTContext &Context = getASTContext();
|
|
|
|
CanQualType ParamType = Context.getCanonicalType(Param->getType());
|
|
|
|
|
|
|
|
// Is it the same as our our class type?
|
|
|
|
CanQualType ClassTy
|
|
|
|
= Context.getCanonicalType(Context.getTagDeclType(getParent()));
|
|
|
|
if (ParamType.getUnqualifiedType() != ClassTy)
|
|
|
|
return false;
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2011-02-06 03:23:19 +08:00
|
|
|
const CXXConstructorDecl *CXXConstructorDecl::getInheritedConstructor() const {
|
|
|
|
// Hack: we store the inherited constructor in the overridden method table
|
2012-03-10 09:39:01 +08:00
|
|
|
method_iterator It = getASTContext().overridden_methods_begin(this);
|
|
|
|
if (It == getASTContext().overridden_methods_end(this))
|
2011-02-06 03:23:19 +08:00
|
|
|
return 0;
|
|
|
|
|
|
|
|
return cast<CXXConstructorDecl>(*It);
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
CXXConstructorDecl::setInheritedConstructor(const CXXConstructorDecl *BaseCtor){
|
|
|
|
// Hack: we store the inherited constructor in the overridden method table
|
2012-03-10 09:39:01 +08:00
|
|
|
assert(getASTContext().overridden_methods_size(this) == 0 &&
|
|
|
|
"Base ctor already set.");
|
|
|
|
getASTContext().addOverriddenMethod(this, BaseCtor);
|
2011-02-06 03:23:19 +08:00
|
|
|
}
|
|
|
|
|
2011-12-20 10:48:34 +08:00
|
|
|
void CXXDestructorDecl::anchor() { }
|
|
|
|
|
2010-05-08 05:43:38 +08:00
|
|
|
CXXDestructorDecl *
|
2012-01-06 05:55:30 +08:00
|
|
|
CXXDestructorDecl::CreateDeserialized(ASTContext &C, unsigned ID) {
|
|
|
|
void *Mem = AllocateDeserializedDecl(C, ID, sizeof(CXXDestructorDecl));
|
|
|
|
return new (Mem) CXXDestructorDecl(0, SourceLocation(), DeclarationNameInfo(),
|
2010-10-21 08:44:50 +08:00
|
|
|
QualType(), 0, false, false);
|
2010-05-08 05:43:38 +08:00
|
|
|
}
|
|
|
|
|
2008-11-06 04:51:48 +08:00
|
|
|
CXXDestructorDecl *
|
|
|
|
CXXDestructorDecl::Create(ASTContext &C, CXXRecordDecl *RD,
|
2011-03-08 16:55:46 +08:00
|
|
|
SourceLocation StartLoc,
|
2010-08-12 06:01:17 +08:00
|
|
|
const DeclarationNameInfo &NameInfo,
|
2010-10-21 08:44:50 +08:00
|
|
|
QualType T, TypeSourceInfo *TInfo,
|
2011-08-16 05:04:07 +08:00
|
|
|
bool isInline, bool isImplicitlyDeclared) {
|
2010-08-12 06:01:17 +08:00
|
|
|
assert(NameInfo.getName().getNameKind()
|
|
|
|
== DeclarationName::CXXDestructorName &&
|
2008-11-17 22:58:09 +08:00
|
|
|
"Name must refer to a destructor");
|
2011-03-08 16:55:46 +08:00
|
|
|
return new (C) CXXDestructorDecl(RD, StartLoc, NameInfo, T, TInfo, isInline,
|
2010-08-12 06:01:17 +08:00
|
|
|
isImplicitlyDeclared);
|
2008-11-06 04:51:48 +08:00
|
|
|
}
|
|
|
|
|
2011-12-20 10:48:34 +08:00
|
|
|
void CXXConversionDecl::anchor() { }
|
|
|
|
|
2010-05-08 05:43:38 +08:00
|
|
|
CXXConversionDecl *
|
2012-01-06 05:55:30 +08:00
|
|
|
CXXConversionDecl::CreateDeserialized(ASTContext &C, unsigned ID) {
|
|
|
|
void *Mem = AllocateDeserializedDecl(C, ID, sizeof(CXXConversionDecl));
|
|
|
|
return new (Mem) CXXConversionDecl(0, SourceLocation(), DeclarationNameInfo(),
|
|
|
|
QualType(), 0, false, false, false,
|
|
|
|
SourceLocation());
|
2010-05-08 05:43:38 +08:00
|
|
|
}
|
|
|
|
|
2008-11-08 04:08:42 +08:00
|
|
|
CXXConversionDecl *
|
|
|
|
CXXConversionDecl::Create(ASTContext &C, CXXRecordDecl *RD,
|
2011-03-08 16:55:46 +08:00
|
|
|
SourceLocation StartLoc,
|
2010-08-12 06:01:17 +08:00
|
|
|
const DeclarationNameInfo &NameInfo,
|
2009-12-07 10:54:59 +08:00
|
|
|
QualType T, TypeSourceInfo *TInfo,
|
2011-03-09 01:10:18 +08:00
|
|
|
bool isInline, bool isExplicit,
|
2011-08-16 05:04:07 +08:00
|
|
|
bool isConstexpr, SourceLocation EndLocation) {
|
2010-08-12 06:01:17 +08:00
|
|
|
assert(NameInfo.getName().getNameKind()
|
|
|
|
== DeclarationName::CXXConversionFunctionName &&
|
2008-11-17 22:58:09 +08:00
|
|
|
"Name must refer to a conversion function");
|
2011-03-08 16:55:46 +08:00
|
|
|
return new (C) CXXConversionDecl(RD, StartLoc, NameInfo, T, TInfo,
|
2011-08-16 05:04:07 +08:00
|
|
|
isInline, isExplicit, isConstexpr,
|
|
|
|
EndLocation);
|
2008-11-08 04:08:42 +08:00
|
|
|
}
|
|
|
|
|
2012-02-16 09:06:16 +08:00
|
|
|
bool CXXConversionDecl::isLambdaToBlockPointerConversion() const {
|
|
|
|
return isImplicit() && getParent()->isLambda() &&
|
|
|
|
getConversionType()->isBlockPointerType();
|
|
|
|
}
|
|
|
|
|
2011-12-20 10:48:34 +08:00
|
|
|
void LinkageSpecDecl::anchor() { }
|
|
|
|
|
2008-11-05 00:51:42 +08:00
|
|
|
LinkageSpecDecl *LinkageSpecDecl::Create(ASTContext &C,
|
2009-09-09 23:08:12 +08:00
|
|
|
DeclContext *DC,
|
2011-03-09 00:41:52 +08:00
|
|
|
SourceLocation ExternLoc,
|
|
|
|
SourceLocation LangLoc,
|
2011-03-03 22:52:38 +08:00
|
|
|
LanguageIDs Lang,
|
|
|
|
SourceLocation RBraceLoc) {
|
2011-03-09 00:41:52 +08:00
|
|
|
return new (C) LinkageSpecDecl(DC, ExternLoc, LangLoc, Lang, RBraceLoc);
|
2008-12-17 06:23:02 +08:00
|
|
|
}
|
2009-02-04 03:21:40 +08:00
|
|
|
|
2012-01-06 05:55:30 +08:00
|
|
|
LinkageSpecDecl *LinkageSpecDecl::CreateDeserialized(ASTContext &C, unsigned ID) {
|
|
|
|
void *Mem = AllocateDeserializedDecl(C, ID, sizeof(LinkageSpecDecl));
|
|
|
|
return new (Mem) LinkageSpecDecl(0, SourceLocation(), SourceLocation(),
|
|
|
|
lang_c, SourceLocation());
|
|
|
|
}
|
|
|
|
|
2011-12-20 10:48:34 +08:00
|
|
|
void UsingDirectiveDecl::anchor() { }
|
|
|
|
|
2009-02-04 03:21:40 +08:00
|
|
|
UsingDirectiveDecl *UsingDirectiveDecl::Create(ASTContext &C, DeclContext *DC,
|
|
|
|
SourceLocation L,
|
|
|
|
SourceLocation NamespaceLoc,
|
2011-02-26 00:33:46 +08:00
|
|
|
NestedNameSpecifierLoc QualifierLoc,
|
2009-02-04 03:21:40 +08:00
|
|
|
SourceLocation IdentLoc,
|
2009-11-23 23:34:23 +08:00
|
|
|
NamedDecl *Used,
|
2009-02-04 03:21:40 +08:00
|
|
|
DeclContext *CommonAncestor) {
|
2009-11-23 23:34:23 +08:00
|
|
|
if (NamespaceDecl *NS = dyn_cast_or_null<NamespaceDecl>(Used))
|
|
|
|
Used = NS->getOriginalNamespace();
|
2011-02-26 00:33:46 +08:00
|
|
|
return new (C) UsingDirectiveDecl(DC, L, NamespaceLoc, QualifierLoc,
|
|
|
|
IdentLoc, Used, CommonAncestor);
|
2009-02-04 03:21:40 +08:00
|
|
|
}
|
|
|
|
|
2012-01-06 05:55:30 +08:00
|
|
|
UsingDirectiveDecl *
|
|
|
|
UsingDirectiveDecl::CreateDeserialized(ASTContext &C, unsigned ID) {
|
|
|
|
void *Mem = AllocateDeserializedDecl(C, ID, sizeof(UsingDirectiveDecl));
|
|
|
|
return new (Mem) UsingDirectiveDecl(0, SourceLocation(), SourceLocation(),
|
|
|
|
NestedNameSpecifierLoc(),
|
|
|
|
SourceLocation(), 0, 0);
|
|
|
|
}
|
|
|
|
|
2009-11-23 23:34:23 +08:00
|
|
|
NamespaceDecl *UsingDirectiveDecl::getNominatedNamespace() {
|
|
|
|
if (NamespaceAliasDecl *NA =
|
|
|
|
dyn_cast_or_null<NamespaceAliasDecl>(NominatedNamespace))
|
|
|
|
return NA->getNamespace();
|
|
|
|
return cast_or_null<NamespaceDecl>(NominatedNamespace);
|
|
|
|
}
|
|
|
|
|
2012-01-06 05:55:30 +08:00
|
|
|
void NamespaceDecl::anchor() { }
|
|
|
|
|
2012-01-07 17:11:48 +08:00
|
|
|
NamespaceDecl::NamespaceDecl(DeclContext *DC, bool Inline,
|
|
|
|
SourceLocation StartLoc,
|
|
|
|
SourceLocation IdLoc, IdentifierInfo *Id,
|
|
|
|
NamespaceDecl *PrevDecl)
|
|
|
|
: NamedDecl(Namespace, DC, IdLoc, Id), DeclContext(Namespace),
|
|
|
|
LocStart(StartLoc), RBraceLoc(), AnonOrFirstNamespaceAndInline(0, Inline)
|
|
|
|
{
|
|
|
|
setPreviousDeclaration(PrevDecl);
|
|
|
|
|
|
|
|
if (PrevDecl)
|
|
|
|
AnonOrFirstNamespaceAndInline.setPointer(PrevDecl->getOriginalNamespace());
|
|
|
|
}
|
|
|
|
|
2012-01-06 05:55:30 +08:00
|
|
|
NamespaceDecl *NamespaceDecl::Create(ASTContext &C, DeclContext *DC,
|
2012-01-07 17:11:48 +08:00
|
|
|
bool Inline, SourceLocation StartLoc,
|
|
|
|
SourceLocation IdLoc, IdentifierInfo *Id,
|
|
|
|
NamespaceDecl *PrevDecl) {
|
|
|
|
return new (C) NamespaceDecl(DC, Inline, StartLoc, IdLoc, Id, PrevDecl);
|
2012-01-06 05:55:30 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
NamespaceDecl *NamespaceDecl::CreateDeserialized(ASTContext &C, unsigned ID) {
|
|
|
|
void *Mem = AllocateDeserializedDecl(C, ID, sizeof(NamespaceDecl));
|
2012-01-07 17:11:48 +08:00
|
|
|
return new (Mem) NamespaceDecl(0, false, SourceLocation(), SourceLocation(),
|
|
|
|
0, 0);
|
2012-01-06 05:55:30 +08:00
|
|
|
}
|
|
|
|
|
2011-12-20 10:48:34 +08:00
|
|
|
void NamespaceAliasDecl::anchor() { }
|
|
|
|
|
2009-09-09 23:08:12 +08:00
|
|
|
NamespaceAliasDecl *NamespaceAliasDecl::Create(ASTContext &C, DeclContext *DC,
|
2010-09-01 11:07:18 +08:00
|
|
|
SourceLocation UsingLoc,
|
2009-09-09 23:08:12 +08:00
|
|
|
SourceLocation AliasLoc,
|
|
|
|
IdentifierInfo *Alias,
|
2011-02-26 01:08:07 +08:00
|
|
|
NestedNameSpecifierLoc QualifierLoc,
|
2009-09-09 23:08:12 +08:00
|
|
|
SourceLocation IdentLoc,
|
2009-03-29 06:58:02 +08:00
|
|
|
NamedDecl *Namespace) {
|
2009-11-23 23:34:23 +08:00
|
|
|
if (NamespaceDecl *NS = dyn_cast_or_null<NamespaceDecl>(Namespace))
|
|
|
|
Namespace = NS->getOriginalNamespace();
|
2011-02-26 01:08:07 +08:00
|
|
|
return new (C) NamespaceAliasDecl(DC, UsingLoc, AliasLoc, Alias,
|
|
|
|
QualifierLoc, IdentLoc, Namespace);
|
2009-03-29 06:58:02 +08:00
|
|
|
}
|
|
|
|
|
2012-01-06 05:55:30 +08:00
|
|
|
NamespaceAliasDecl *
|
|
|
|
NamespaceAliasDecl::CreateDeserialized(ASTContext &C, unsigned ID) {
|
|
|
|
void *Mem = AllocateDeserializedDecl(C, ID, sizeof(NamespaceAliasDecl));
|
|
|
|
return new (Mem) NamespaceAliasDecl(0, SourceLocation(), SourceLocation(), 0,
|
|
|
|
NestedNameSpecifierLoc(),
|
|
|
|
SourceLocation(), 0);
|
|
|
|
}
|
|
|
|
|
2011-12-20 10:48:34 +08:00
|
|
|
void UsingShadowDecl::anchor() { }
|
|
|
|
|
2012-01-06 05:55:30 +08:00
|
|
|
UsingShadowDecl *
|
|
|
|
UsingShadowDecl::CreateDeserialized(ASTContext &C, unsigned ID) {
|
|
|
|
void *Mem = AllocateDeserializedDecl(C, ID, sizeof(UsingShadowDecl));
|
|
|
|
return new (Mem) UsingShadowDecl(0, SourceLocation(), 0, 0);
|
|
|
|
}
|
|
|
|
|
2010-11-10 13:40:41 +08:00
|
|
|
UsingDecl *UsingShadowDecl::getUsingDecl() const {
|
|
|
|
const UsingShadowDecl *Shadow = this;
|
|
|
|
while (const UsingShadowDecl *NextShadow =
|
|
|
|
dyn_cast<UsingShadowDecl>(Shadow->UsingOrNextShadow))
|
|
|
|
Shadow = NextShadow;
|
|
|
|
return cast<UsingDecl>(Shadow->UsingOrNextShadow);
|
|
|
|
}
|
|
|
|
|
2011-12-20 10:48:34 +08:00
|
|
|
void UsingDecl::anchor() { }
|
|
|
|
|
2010-11-10 13:40:41 +08:00
|
|
|
void UsingDecl::addShadowDecl(UsingShadowDecl *S) {
|
|
|
|
assert(std::find(shadow_begin(), shadow_end(), S) == shadow_end() &&
|
|
|
|
"declaration already in set");
|
|
|
|
assert(S->getUsingDecl() == this);
|
|
|
|
|
2012-01-08 03:09:05 +08:00
|
|
|
if (FirstUsingShadow.getPointer())
|
|
|
|
S->UsingOrNextShadow = FirstUsingShadow.getPointer();
|
|
|
|
FirstUsingShadow.setPointer(S);
|
2010-11-10 13:40:41 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
void UsingDecl::removeShadowDecl(UsingShadowDecl *S) {
|
|
|
|
assert(std::find(shadow_begin(), shadow_end(), S) != shadow_end() &&
|
|
|
|
"declaration not in set");
|
|
|
|
assert(S->getUsingDecl() == this);
|
|
|
|
|
|
|
|
// Remove S from the shadow decl chain. This is O(n) but hopefully rare.
|
|
|
|
|
2012-01-08 03:09:05 +08:00
|
|
|
if (FirstUsingShadow.getPointer() == S) {
|
|
|
|
FirstUsingShadow.setPointer(
|
|
|
|
dyn_cast<UsingShadowDecl>(S->UsingOrNextShadow));
|
2010-11-10 13:40:41 +08:00
|
|
|
S->UsingOrNextShadow = this;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2012-01-08 03:09:05 +08:00
|
|
|
UsingShadowDecl *Prev = FirstUsingShadow.getPointer();
|
2010-11-10 13:40:41 +08:00
|
|
|
while (Prev->UsingOrNextShadow != S)
|
|
|
|
Prev = cast<UsingShadowDecl>(Prev->UsingOrNextShadow);
|
|
|
|
Prev->UsingOrNextShadow = S->UsingOrNextShadow;
|
|
|
|
S->UsingOrNextShadow = this;
|
|
|
|
}
|
|
|
|
|
2011-02-25 08:36:19 +08:00
|
|
|
UsingDecl *UsingDecl::Create(ASTContext &C, DeclContext *DC, SourceLocation UL,
|
|
|
|
NestedNameSpecifierLoc QualifierLoc,
|
2010-08-12 19:46:03 +08:00
|
|
|
const DeclarationNameInfo &NameInfo,
|
|
|
|
bool IsTypeNameArg) {
|
2011-02-25 08:36:19 +08:00
|
|
|
return new (C) UsingDecl(DC, UL, QualifierLoc, NameInfo, IsTypeNameArg);
|
2009-06-20 08:51:54 +08:00
|
|
|
}
|
|
|
|
|
2012-01-06 05:55:30 +08:00
|
|
|
UsingDecl *UsingDecl::CreateDeserialized(ASTContext &C, unsigned ID) {
|
|
|
|
void *Mem = AllocateDeserializedDecl(C, ID, sizeof(UsingDecl));
|
|
|
|
return new (Mem) UsingDecl(0, SourceLocation(), NestedNameSpecifierLoc(),
|
|
|
|
DeclarationNameInfo(), false);
|
|
|
|
}
|
|
|
|
|
2011-12-20 10:48:34 +08:00
|
|
|
void UnresolvedUsingValueDecl::anchor() { }
|
|
|
|
|
2009-11-18 10:36:19 +08:00
|
|
|
UnresolvedUsingValueDecl *
|
|
|
|
UnresolvedUsingValueDecl::Create(ASTContext &C, DeclContext *DC,
|
|
|
|
SourceLocation UsingLoc,
|
2011-02-25 08:36:19 +08:00
|
|
|
NestedNameSpecifierLoc QualifierLoc,
|
2010-08-12 19:46:03 +08:00
|
|
|
const DeclarationNameInfo &NameInfo) {
|
2009-11-18 10:36:19 +08:00
|
|
|
return new (C) UnresolvedUsingValueDecl(DC, C.DependentTy, UsingLoc,
|
2011-02-25 08:36:19 +08:00
|
|
|
QualifierLoc, NameInfo);
|
2009-11-18 10:36:19 +08:00
|
|
|
}
|
|
|
|
|
2012-01-06 05:55:30 +08:00
|
|
|
UnresolvedUsingValueDecl *
|
|
|
|
UnresolvedUsingValueDecl::CreateDeserialized(ASTContext &C, unsigned ID) {
|
|
|
|
void *Mem = AllocateDeserializedDecl(C, ID, sizeof(UnresolvedUsingValueDecl));
|
|
|
|
return new (Mem) UnresolvedUsingValueDecl(0, QualType(), SourceLocation(),
|
|
|
|
NestedNameSpecifierLoc(),
|
|
|
|
DeclarationNameInfo());
|
|
|
|
}
|
|
|
|
|
2011-12-20 10:48:34 +08:00
|
|
|
void UnresolvedUsingTypenameDecl::anchor() { }
|
|
|
|
|
2009-11-18 10:36:19 +08:00
|
|
|
UnresolvedUsingTypenameDecl *
|
|
|
|
UnresolvedUsingTypenameDecl::Create(ASTContext &C, DeclContext *DC,
|
|
|
|
SourceLocation UsingLoc,
|
|
|
|
SourceLocation TypenameLoc,
|
2011-02-25 08:36:19 +08:00
|
|
|
NestedNameSpecifierLoc QualifierLoc,
|
2009-11-18 10:36:19 +08:00
|
|
|
SourceLocation TargetNameLoc,
|
|
|
|
DeclarationName TargetName) {
|
|
|
|
return new (C) UnresolvedUsingTypenameDecl(DC, UsingLoc, TypenameLoc,
|
2011-02-25 08:36:19 +08:00
|
|
|
QualifierLoc, TargetNameLoc,
|
2009-11-18 10:36:19 +08:00
|
|
|
TargetName.getAsIdentifierInfo());
|
2009-08-28 13:30:28 +08:00
|
|
|
}
|
|
|
|
|
2012-01-06 05:55:30 +08:00
|
|
|
UnresolvedUsingTypenameDecl *
|
|
|
|
UnresolvedUsingTypenameDecl::CreateDeserialized(ASTContext &C, unsigned ID) {
|
|
|
|
void *Mem = AllocateDeserializedDecl(C, ID,
|
|
|
|
sizeof(UnresolvedUsingTypenameDecl));
|
|
|
|
return new (Mem) UnresolvedUsingTypenameDecl(0, SourceLocation(),
|
|
|
|
SourceLocation(),
|
|
|
|
NestedNameSpecifierLoc(),
|
|
|
|
SourceLocation(),
|
|
|
|
0);
|
|
|
|
}
|
|
|
|
|
2011-12-20 10:48:34 +08:00
|
|
|
void StaticAssertDecl::anchor() { }
|
|
|
|
|
2009-03-14 08:25:26 +08:00
|
|
|
StaticAssertDecl *StaticAssertDecl::Create(ASTContext &C, DeclContext *DC,
|
2011-03-09 00:41:52 +08:00
|
|
|
SourceLocation StaticAssertLoc,
|
|
|
|
Expr *AssertExpr,
|
|
|
|
StringLiteral *Message,
|
|
|
|
SourceLocation RParenLoc) {
|
|
|
|
return new (C) StaticAssertDecl(DC, StaticAssertLoc, AssertExpr, Message,
|
|
|
|
RParenLoc);
|
2009-03-14 08:25:26 +08:00
|
|
|
}
|
|
|
|
|
2012-01-06 05:55:30 +08:00
|
|
|
StaticAssertDecl *StaticAssertDecl::CreateDeserialized(ASTContext &C,
|
|
|
|
unsigned ID) {
|
|
|
|
void *Mem = AllocateDeserializedDecl(C, ID, sizeof(StaticAssertDecl));
|
|
|
|
return new (Mem) StaticAssertDecl(0, SourceLocation(), 0, 0,SourceLocation());
|
|
|
|
}
|
|
|
|
|
2009-03-27 07:46:50 +08:00
|
|
|
static const char *getAccessName(AccessSpecifier AS) {
|
|
|
|
switch (AS) {
|
|
|
|
case AS_none:
|
2011-09-23 13:06:16 +08:00
|
|
|
llvm_unreachable("Invalid access specifier!");
|
2009-03-27 07:46:50 +08:00
|
|
|
case AS_public:
|
|
|
|
return "public";
|
|
|
|
case AS_private:
|
|
|
|
return "private";
|
|
|
|
case AS_protected:
|
|
|
|
return "protected";
|
|
|
|
}
|
2012-01-17 10:30:50 +08:00
|
|
|
llvm_unreachable("Invalid access specifier!");
|
2009-03-27 07:46:50 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
const DiagnosticBuilder &clang::operator<<(const DiagnosticBuilder &DB,
|
|
|
|
AccessSpecifier AS) {
|
|
|
|
return DB << getAccessName(AS);
|
|
|
|
}
|
2012-02-02 09:16:57 +08:00
|
|
|
|
|
|
|
const PartialDiagnostic &clang::operator<<(const PartialDiagnostic &DB,
|
|
|
|
AccessSpecifier AS) {
|
|
|
|
return DB << getAccessName(AS);
|
|
|
|
}
|