2006-10-16 06:34:45 +08:00
|
|
|
//===--- Decl.cpp - Declaration AST Node Implementation -------------------===//
|
|
|
|
//
|
|
|
|
// The LLVM Compiler Infrastructure
|
|
|
|
//
|
2007-12-30 03:59:25 +08:00
|
|
|
// This file is distributed under the University of Illinois Open Source
|
|
|
|
// License. See LICENSE.TXT for details.
|
2006-10-16 06:34:45 +08:00
|
|
|
//
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
//
|
2008-06-04 21:04:04 +08:00
|
|
|
// This file implements the Decl subclasses.
|
2006-10-16 06:34:45 +08:00
|
|
|
//
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
|
|
|
|
#include "clang/AST/Decl.h"
|
2009-02-04 03:21:40 +08:00
|
|
|
#include "clang/AST/DeclCXX.h"
|
2009-02-23 03:35:57 +08:00
|
|
|
#include "clang/AST/DeclObjC.h"
|
2009-05-11 06:57:19 +08:00
|
|
|
#include "clang/AST/DeclTemplate.h"
|
2008-03-15 14:12:44 +08:00
|
|
|
#include "clang/AST/ASTContext.h"
|
2009-08-19 09:27:32 +08:00
|
|
|
#include "clang/AST/TypeLoc.h"
|
2008-08-11 12:54:23 +08:00
|
|
|
#include "clang/AST/Stmt.h"
|
2008-12-18 07:39:55 +08:00
|
|
|
#include "clang/AST/Expr.h"
|
2009-12-16 03:16:31 +08:00
|
|
|
#include "clang/AST/ExprCXX.h"
|
2009-05-30 04:38:28 +08:00
|
|
|
#include "clang/AST/PrettyPrinter.h"
|
2009-06-14 09:54:56 +08:00
|
|
|
#include "clang/Basic/Builtins.h"
|
2008-08-11 12:54:23 +08:00
|
|
|
#include "clang/Basic/IdentifierTable.h"
|
2009-09-04 09:14:41 +08:00
|
|
|
#include "clang/Parse/DeclSpec.h"
|
|
|
|
#include "llvm/Support/ErrorHandling.h"
|
2009-02-05 01:27:36 +08:00
|
|
|
#include <vector>
|
2008-05-20 08:43:19 +08:00
|
|
|
|
2006-10-25 13:11:20 +08:00
|
|
|
using namespace clang;
|
2006-10-16 06:34:45 +08:00
|
|
|
|
2009-08-19 09:27:32 +08:00
|
|
|
/// \brief Return the TypeLoc wrapper for the type source info.
|
2009-12-07 10:54:59 +08:00
|
|
|
TypeLoc TypeSourceInfo::getTypeLoc() const {
|
2009-09-30 03:40:20 +08:00
|
|
|
return TypeLoc(Ty, (void*)(this + 1));
|
2009-08-19 09:27:32 +08:00
|
|
|
}
|
2009-03-04 14:34:08 +08:00
|
|
|
|
2008-03-31 08:36:02 +08:00
|
|
|
//===----------------------------------------------------------------------===//
|
2009-01-20 09:17:11 +08:00
|
|
|
// NamedDecl Implementation
|
2008-11-10 07:41:00 +08:00
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
|
When a function or variable somehow depends on a type or declaration
that is in an anonymous namespace, give that function or variable
internal linkage.
This change models an oddity of the C++ standard, where names declared
in an anonymous namespace have external linkage but, because anonymous
namespace are really "uniquely-named" namespaces, the names cannot be
referenced from other translation units. That means that they have
external linkage for semantic analysis, but the only sensible
implementation for code generation is to give them internal
linkage. We now model this notion via the UniqueExternalLinkage
linkage type. There are several changes here:
- Extended NamedDecl::getLinkage() to produce UniqueExternalLinkage
when the declaration is in an anonymous namespace.
- Added Type::getLinkage() to determine the linkage of a type, which
is defined as the minimum linkage of the types (when we're dealing
with a compound type that is not a struct/class/union).
- Extended NamedDecl::getLinkage() to consider the linkage of the
template arguments and template parameters of function template
specializations and class template specializations.
- Taught code generation to rely on NamedDecl::getLinkage() when
determining the linkage of variables and functions, also
considering the linkage of the types of those variables and
functions (C++ only). Map UniqueExternalLinkage to internal
linkage, taking out the explicit checks for
isInAnonymousNamespace().
This fixes much of PR5792, which, as discovered by Anders Carlsson, is
actually the reason behind the pass-manager assertion that causes the
majority of clang-on-clang regression test failures. With this fix,
Clang-built-Clang+LLVM passes 88% of its regression tests (up from
67%). The specific numbers are:
LLVM:
Expected Passes : 4006
Expected Failures : 32
Unsupported Tests : 40
Unexpected Failures: 736
Clang:
Expected Passes : 1903
Expected Failures : 14
Unexpected Failures: 75
Overall:
Expected Passes : 5909
Expected Failures : 46
Unsupported Tests : 40
Unexpected Failures: 811
Still to do:
- Improve testing
- Check whether we should allow the presence of types with
InternalLinkage (in addition to UniqueExternalLinkage) given
variables/functions internal linkage in C++, as mentioned in
PR5792.
- Determine how expensive the getLinkage() calls are in practice;
consider caching the result in NamedDecl.
- Assess the feasibility of Chris's idea in comment #1 of PR5792.
llvm-svn: 95216
2010-02-03 17:33:45 +08:00
|
|
|
/// \brief Get the most restrictive linkage for the types in the given
|
|
|
|
/// template parameter list.
|
|
|
|
static Linkage
|
|
|
|
getLinkageForTemplateParameterList(const TemplateParameterList *Params) {
|
|
|
|
Linkage L = ExternalLinkage;
|
|
|
|
for (TemplateParameterList::const_iterator P = Params->begin(),
|
|
|
|
PEnd = Params->end();
|
|
|
|
P != PEnd; ++P) {
|
|
|
|
if (NonTypeTemplateParmDecl *NTTP = dyn_cast<NonTypeTemplateParmDecl>(*P))
|
|
|
|
if (!NTTP->getType()->isDependentType()) {
|
|
|
|
L = minLinkage(L, NTTP->getType()->getLinkage());
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (TemplateTemplateParmDecl *TTP
|
|
|
|
= dyn_cast<TemplateTemplateParmDecl>(*P)) {
|
|
|
|
L = minLinkage(L,
|
|
|
|
getLinkageForTemplateParameterList(TTP->getTemplateParameters()));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return L;
|
|
|
|
}
|
|
|
|
|
|
|
|
/// \brief Get the most restrictive linkage for the types and
|
|
|
|
/// declarations in the given template argument list.
|
|
|
|
static Linkage getLinkageForTemplateArgumentList(const TemplateArgument *Args,
|
|
|
|
unsigned NumArgs) {
|
|
|
|
Linkage L = ExternalLinkage;
|
|
|
|
|
|
|
|
for (unsigned I = 0; I != NumArgs; ++I) {
|
|
|
|
switch (Args[I].getKind()) {
|
|
|
|
case TemplateArgument::Null:
|
|
|
|
case TemplateArgument::Integral:
|
|
|
|
case TemplateArgument::Expression:
|
|
|
|
break;
|
|
|
|
|
|
|
|
case TemplateArgument::Type:
|
|
|
|
L = minLinkage(L, Args[I].getAsType()->getLinkage());
|
|
|
|
break;
|
|
|
|
|
|
|
|
case TemplateArgument::Declaration:
|
|
|
|
if (NamedDecl *ND = dyn_cast<NamedDecl>(Args[I].getAsDecl()))
|
|
|
|
L = minLinkage(L, ND->getLinkage());
|
|
|
|
if (ValueDecl *VD = dyn_cast<ValueDecl>(Args[I].getAsDecl()))
|
|
|
|
L = minLinkage(L, VD->getType()->getLinkage());
|
|
|
|
break;
|
|
|
|
|
|
|
|
case TemplateArgument::Template:
|
|
|
|
if (TemplateDecl *Template
|
|
|
|
= Args[I].getAsTemplate().getAsTemplateDecl())
|
|
|
|
L = minLinkage(L, Template->getLinkage());
|
|
|
|
break;
|
|
|
|
|
|
|
|
case TemplateArgument::Pack:
|
|
|
|
L = minLinkage(L,
|
|
|
|
getLinkageForTemplateArgumentList(Args[I].pack_begin(),
|
|
|
|
Args[I].pack_size()));
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return L;
|
|
|
|
}
|
|
|
|
|
|
|
|
static Linkage getLinkageForNamespaceScopeDecl(const NamedDecl *D) {
|
2009-11-26 06:24:25 +08:00
|
|
|
assert(D->getDeclContext()->getLookupContext()->isFileContext() &&
|
|
|
|
"Not a name having namespace scope");
|
|
|
|
ASTContext &Context = D->getASTContext();
|
|
|
|
|
|
|
|
// C++ [basic.link]p3:
|
|
|
|
// A name having namespace scope (3.3.6) has internal linkage if it
|
|
|
|
// is the name of
|
|
|
|
// - an object, reference, function or function template that is
|
|
|
|
// explicitly declared static; or,
|
|
|
|
// (This bullet corresponds to C99 6.2.2p3.)
|
|
|
|
if (const VarDecl *Var = dyn_cast<VarDecl>(D)) {
|
|
|
|
// Explicitly declared static.
|
|
|
|
if (Var->getStorageClass() == VarDecl::Static)
|
When a function or variable somehow depends on a type or declaration
that is in an anonymous namespace, give that function or variable
internal linkage.
This change models an oddity of the C++ standard, where names declared
in an anonymous namespace have external linkage but, because anonymous
namespace are really "uniquely-named" namespaces, the names cannot be
referenced from other translation units. That means that they have
external linkage for semantic analysis, but the only sensible
implementation for code generation is to give them internal
linkage. We now model this notion via the UniqueExternalLinkage
linkage type. There are several changes here:
- Extended NamedDecl::getLinkage() to produce UniqueExternalLinkage
when the declaration is in an anonymous namespace.
- Added Type::getLinkage() to determine the linkage of a type, which
is defined as the minimum linkage of the types (when we're dealing
with a compound type that is not a struct/class/union).
- Extended NamedDecl::getLinkage() to consider the linkage of the
template arguments and template parameters of function template
specializations and class template specializations.
- Taught code generation to rely on NamedDecl::getLinkage() when
determining the linkage of variables and functions, also
considering the linkage of the types of those variables and
functions (C++ only). Map UniqueExternalLinkage to internal
linkage, taking out the explicit checks for
isInAnonymousNamespace().
This fixes much of PR5792, which, as discovered by Anders Carlsson, is
actually the reason behind the pass-manager assertion that causes the
majority of clang-on-clang regression test failures. With this fix,
Clang-built-Clang+LLVM passes 88% of its regression tests (up from
67%). The specific numbers are:
LLVM:
Expected Passes : 4006
Expected Failures : 32
Unsupported Tests : 40
Unexpected Failures: 736
Clang:
Expected Passes : 1903
Expected Failures : 14
Unexpected Failures: 75
Overall:
Expected Passes : 5909
Expected Failures : 46
Unsupported Tests : 40
Unexpected Failures: 811
Still to do:
- Improve testing
- Check whether we should allow the presence of types with
InternalLinkage (in addition to UniqueExternalLinkage) given
variables/functions internal linkage in C++, as mentioned in
PR5792.
- Determine how expensive the getLinkage() calls are in practice;
consider caching the result in NamedDecl.
- Assess the feasibility of Chris's idea in comment #1 of PR5792.
llvm-svn: 95216
2010-02-03 17:33:45 +08:00
|
|
|
return InternalLinkage;
|
2009-11-26 06:24:25 +08:00
|
|
|
|
|
|
|
// - an object or reference that is explicitly declared const
|
|
|
|
// and neither explicitly declared extern nor previously
|
|
|
|
// declared to have external linkage; or
|
|
|
|
// (there is no equivalent in C99)
|
|
|
|
if (Context.getLangOptions().CPlusPlus &&
|
2009-11-26 11:04:01 +08:00
|
|
|
Var->getType().isConstant(Context) &&
|
2009-11-26 06:24:25 +08:00
|
|
|
Var->getStorageClass() != VarDecl::Extern &&
|
|
|
|
Var->getStorageClass() != VarDecl::PrivateExtern) {
|
|
|
|
bool FoundExtern = false;
|
|
|
|
for (const VarDecl *PrevVar = Var->getPreviousDeclaration();
|
|
|
|
PrevVar && !FoundExtern;
|
|
|
|
PrevVar = PrevVar->getPreviousDeclaration())
|
When a function or variable somehow depends on a type or declaration
that is in an anonymous namespace, give that function or variable
internal linkage.
This change models an oddity of the C++ standard, where names declared
in an anonymous namespace have external linkage but, because anonymous
namespace are really "uniquely-named" namespaces, the names cannot be
referenced from other translation units. That means that they have
external linkage for semantic analysis, but the only sensible
implementation for code generation is to give them internal
linkage. We now model this notion via the UniqueExternalLinkage
linkage type. There are several changes here:
- Extended NamedDecl::getLinkage() to produce UniqueExternalLinkage
when the declaration is in an anonymous namespace.
- Added Type::getLinkage() to determine the linkage of a type, which
is defined as the minimum linkage of the types (when we're dealing
with a compound type that is not a struct/class/union).
- Extended NamedDecl::getLinkage() to consider the linkage of the
template arguments and template parameters of function template
specializations and class template specializations.
- Taught code generation to rely on NamedDecl::getLinkage() when
determining the linkage of variables and functions, also
considering the linkage of the types of those variables and
functions (C++ only). Map UniqueExternalLinkage to internal
linkage, taking out the explicit checks for
isInAnonymousNamespace().
This fixes much of PR5792, which, as discovered by Anders Carlsson, is
actually the reason behind the pass-manager assertion that causes the
majority of clang-on-clang regression test failures. With this fix,
Clang-built-Clang+LLVM passes 88% of its regression tests (up from
67%). The specific numbers are:
LLVM:
Expected Passes : 4006
Expected Failures : 32
Unsupported Tests : 40
Unexpected Failures: 736
Clang:
Expected Passes : 1903
Expected Failures : 14
Unexpected Failures: 75
Overall:
Expected Passes : 5909
Expected Failures : 46
Unsupported Tests : 40
Unexpected Failures: 811
Still to do:
- Improve testing
- Check whether we should allow the presence of types with
InternalLinkage (in addition to UniqueExternalLinkage) given
variables/functions internal linkage in C++, as mentioned in
PR5792.
- Determine how expensive the getLinkage() calls are in practice;
consider caching the result in NamedDecl.
- Assess the feasibility of Chris's idea in comment #1 of PR5792.
llvm-svn: 95216
2010-02-03 17:33:45 +08:00
|
|
|
if (isExternalLinkage(PrevVar->getLinkage()))
|
2009-11-26 06:24:25 +08:00
|
|
|
FoundExtern = true;
|
|
|
|
|
|
|
|
if (!FoundExtern)
|
When a function or variable somehow depends on a type or declaration
that is in an anonymous namespace, give that function or variable
internal linkage.
This change models an oddity of the C++ standard, where names declared
in an anonymous namespace have external linkage but, because anonymous
namespace are really "uniquely-named" namespaces, the names cannot be
referenced from other translation units. That means that they have
external linkage for semantic analysis, but the only sensible
implementation for code generation is to give them internal
linkage. We now model this notion via the UniqueExternalLinkage
linkage type. There are several changes here:
- Extended NamedDecl::getLinkage() to produce UniqueExternalLinkage
when the declaration is in an anonymous namespace.
- Added Type::getLinkage() to determine the linkage of a type, which
is defined as the minimum linkage of the types (when we're dealing
with a compound type that is not a struct/class/union).
- Extended NamedDecl::getLinkage() to consider the linkage of the
template arguments and template parameters of function template
specializations and class template specializations.
- Taught code generation to rely on NamedDecl::getLinkage() when
determining the linkage of variables and functions, also
considering the linkage of the types of those variables and
functions (C++ only). Map UniqueExternalLinkage to internal
linkage, taking out the explicit checks for
isInAnonymousNamespace().
This fixes much of PR5792, which, as discovered by Anders Carlsson, is
actually the reason behind the pass-manager assertion that causes the
majority of clang-on-clang regression test failures. With this fix,
Clang-built-Clang+LLVM passes 88% of its regression tests (up from
67%). The specific numbers are:
LLVM:
Expected Passes : 4006
Expected Failures : 32
Unsupported Tests : 40
Unexpected Failures: 736
Clang:
Expected Passes : 1903
Expected Failures : 14
Unexpected Failures: 75
Overall:
Expected Passes : 5909
Expected Failures : 46
Unsupported Tests : 40
Unexpected Failures: 811
Still to do:
- Improve testing
- Check whether we should allow the presence of types with
InternalLinkage (in addition to UniqueExternalLinkage) given
variables/functions internal linkage in C++, as mentioned in
PR5792.
- Determine how expensive the getLinkage() calls are in practice;
consider caching the result in NamedDecl.
- Assess the feasibility of Chris's idea in comment #1 of PR5792.
llvm-svn: 95216
2010-02-03 17:33:45 +08:00
|
|
|
return InternalLinkage;
|
2009-11-26 06:24:25 +08:00
|
|
|
}
|
|
|
|
} else if (isa<FunctionDecl>(D) || isa<FunctionTemplateDecl>(D)) {
|
When a function or variable somehow depends on a type or declaration
that is in an anonymous namespace, give that function or variable
internal linkage.
This change models an oddity of the C++ standard, where names declared
in an anonymous namespace have external linkage but, because anonymous
namespace are really "uniquely-named" namespaces, the names cannot be
referenced from other translation units. That means that they have
external linkage for semantic analysis, but the only sensible
implementation for code generation is to give them internal
linkage. We now model this notion via the UniqueExternalLinkage
linkage type. There are several changes here:
- Extended NamedDecl::getLinkage() to produce UniqueExternalLinkage
when the declaration is in an anonymous namespace.
- Added Type::getLinkage() to determine the linkage of a type, which
is defined as the minimum linkage of the types (when we're dealing
with a compound type that is not a struct/class/union).
- Extended NamedDecl::getLinkage() to consider the linkage of the
template arguments and template parameters of function template
specializations and class template specializations.
- Taught code generation to rely on NamedDecl::getLinkage() when
determining the linkage of variables and functions, also
considering the linkage of the types of those variables and
functions (C++ only). Map UniqueExternalLinkage to internal
linkage, taking out the explicit checks for
isInAnonymousNamespace().
This fixes much of PR5792, which, as discovered by Anders Carlsson, is
actually the reason behind the pass-manager assertion that causes the
majority of clang-on-clang regression test failures. With this fix,
Clang-built-Clang+LLVM passes 88% of its regression tests (up from
67%). The specific numbers are:
LLVM:
Expected Passes : 4006
Expected Failures : 32
Unsupported Tests : 40
Unexpected Failures: 736
Clang:
Expected Passes : 1903
Expected Failures : 14
Unexpected Failures: 75
Overall:
Expected Passes : 5909
Expected Failures : 46
Unsupported Tests : 40
Unexpected Failures: 811
Still to do:
- Improve testing
- Check whether we should allow the presence of types with
InternalLinkage (in addition to UniqueExternalLinkage) given
variables/functions internal linkage in C++, as mentioned in
PR5792.
- Determine how expensive the getLinkage() calls are in practice;
consider caching the result in NamedDecl.
- Assess the feasibility of Chris's idea in comment #1 of PR5792.
llvm-svn: 95216
2010-02-03 17:33:45 +08:00
|
|
|
// C++ [temp]p4:
|
|
|
|
// A non-member function template can have internal linkage; any
|
|
|
|
// other template name shall have external linkage.
|
2009-11-26 06:24:25 +08:00
|
|
|
const FunctionDecl *Function = 0;
|
|
|
|
if (const FunctionTemplateDecl *FunTmpl
|
|
|
|
= dyn_cast<FunctionTemplateDecl>(D))
|
|
|
|
Function = FunTmpl->getTemplatedDecl();
|
|
|
|
else
|
|
|
|
Function = cast<FunctionDecl>(D);
|
|
|
|
|
|
|
|
// Explicitly declared static.
|
|
|
|
if (Function->getStorageClass() == FunctionDecl::Static)
|
When a function or variable somehow depends on a type or declaration
that is in an anonymous namespace, give that function or variable
internal linkage.
This change models an oddity of the C++ standard, where names declared
in an anonymous namespace have external linkage but, because anonymous
namespace are really "uniquely-named" namespaces, the names cannot be
referenced from other translation units. That means that they have
external linkage for semantic analysis, but the only sensible
implementation for code generation is to give them internal
linkage. We now model this notion via the UniqueExternalLinkage
linkage type. There are several changes here:
- Extended NamedDecl::getLinkage() to produce UniqueExternalLinkage
when the declaration is in an anonymous namespace.
- Added Type::getLinkage() to determine the linkage of a type, which
is defined as the minimum linkage of the types (when we're dealing
with a compound type that is not a struct/class/union).
- Extended NamedDecl::getLinkage() to consider the linkage of the
template arguments and template parameters of function template
specializations and class template specializations.
- Taught code generation to rely on NamedDecl::getLinkage() when
determining the linkage of variables and functions, also
considering the linkage of the types of those variables and
functions (C++ only). Map UniqueExternalLinkage to internal
linkage, taking out the explicit checks for
isInAnonymousNamespace().
This fixes much of PR5792, which, as discovered by Anders Carlsson, is
actually the reason behind the pass-manager assertion that causes the
majority of clang-on-clang regression test failures. With this fix,
Clang-built-Clang+LLVM passes 88% of its regression tests (up from
67%). The specific numbers are:
LLVM:
Expected Passes : 4006
Expected Failures : 32
Unsupported Tests : 40
Unexpected Failures: 736
Clang:
Expected Passes : 1903
Expected Failures : 14
Unexpected Failures: 75
Overall:
Expected Passes : 5909
Expected Failures : 46
Unsupported Tests : 40
Unexpected Failures: 811
Still to do:
- Improve testing
- Check whether we should allow the presence of types with
InternalLinkage (in addition to UniqueExternalLinkage) given
variables/functions internal linkage in C++, as mentioned in
PR5792.
- Determine how expensive the getLinkage() calls are in practice;
consider caching the result in NamedDecl.
- Assess the feasibility of Chris's idea in comment #1 of PR5792.
llvm-svn: 95216
2010-02-03 17:33:45 +08:00
|
|
|
return InternalLinkage;
|
2009-11-26 06:24:25 +08:00
|
|
|
} else if (const FieldDecl *Field = dyn_cast<FieldDecl>(D)) {
|
|
|
|
// - a data member of an anonymous union.
|
|
|
|
if (cast<RecordDecl>(Field->getDeclContext())->isAnonymousStructOrUnion())
|
When a function or variable somehow depends on a type or declaration
that is in an anonymous namespace, give that function or variable
internal linkage.
This change models an oddity of the C++ standard, where names declared
in an anonymous namespace have external linkage but, because anonymous
namespace are really "uniquely-named" namespaces, the names cannot be
referenced from other translation units. That means that they have
external linkage for semantic analysis, but the only sensible
implementation for code generation is to give them internal
linkage. We now model this notion via the UniqueExternalLinkage
linkage type. There are several changes here:
- Extended NamedDecl::getLinkage() to produce UniqueExternalLinkage
when the declaration is in an anonymous namespace.
- Added Type::getLinkage() to determine the linkage of a type, which
is defined as the minimum linkage of the types (when we're dealing
with a compound type that is not a struct/class/union).
- Extended NamedDecl::getLinkage() to consider the linkage of the
template arguments and template parameters of function template
specializations and class template specializations.
- Taught code generation to rely on NamedDecl::getLinkage() when
determining the linkage of variables and functions, also
considering the linkage of the types of those variables and
functions (C++ only). Map UniqueExternalLinkage to internal
linkage, taking out the explicit checks for
isInAnonymousNamespace().
This fixes much of PR5792, which, as discovered by Anders Carlsson, is
actually the reason behind the pass-manager assertion that causes the
majority of clang-on-clang regression test failures. With this fix,
Clang-built-Clang+LLVM passes 88% of its regression tests (up from
67%). The specific numbers are:
LLVM:
Expected Passes : 4006
Expected Failures : 32
Unsupported Tests : 40
Unexpected Failures: 736
Clang:
Expected Passes : 1903
Expected Failures : 14
Unexpected Failures: 75
Overall:
Expected Passes : 5909
Expected Failures : 46
Unsupported Tests : 40
Unexpected Failures: 811
Still to do:
- Improve testing
- Check whether we should allow the presence of types with
InternalLinkage (in addition to UniqueExternalLinkage) given
variables/functions internal linkage in C++, as mentioned in
PR5792.
- Determine how expensive the getLinkage() calls are in practice;
consider caching the result in NamedDecl.
- Assess the feasibility of Chris's idea in comment #1 of PR5792.
llvm-svn: 95216
2010-02-03 17:33:45 +08:00
|
|
|
return InternalLinkage;
|
2009-11-26 06:24:25 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
// C++ [basic.link]p4:
|
|
|
|
|
|
|
|
// A name having namespace scope has external linkage if it is the
|
|
|
|
// name of
|
|
|
|
//
|
|
|
|
// - an object or reference, unless it has internal linkage; or
|
|
|
|
if (const VarDecl *Var = dyn_cast<VarDecl>(D)) {
|
|
|
|
if (!Context.getLangOptions().CPlusPlus &&
|
|
|
|
(Var->getStorageClass() == VarDecl::Extern ||
|
|
|
|
Var->getStorageClass() == VarDecl::PrivateExtern)) {
|
|
|
|
// C99 6.2.2p4:
|
|
|
|
// For an identifier declared with the storage-class specifier
|
|
|
|
// extern in a scope in which a prior declaration of that
|
|
|
|
// identifier is visible, if the prior declaration specifies
|
|
|
|
// internal or external linkage, the linkage of the identifier
|
|
|
|
// at the later declaration is the same as the linkage
|
|
|
|
// specified at the prior declaration. If no prior declaration
|
|
|
|
// is visible, or if the prior declaration specifies no
|
|
|
|
// linkage, then the identifier has external linkage.
|
|
|
|
if (const VarDecl *PrevVar = Var->getPreviousDeclaration()) {
|
When a function or variable somehow depends on a type or declaration
that is in an anonymous namespace, give that function or variable
internal linkage.
This change models an oddity of the C++ standard, where names declared
in an anonymous namespace have external linkage but, because anonymous
namespace are really "uniquely-named" namespaces, the names cannot be
referenced from other translation units. That means that they have
external linkage for semantic analysis, but the only sensible
implementation for code generation is to give them internal
linkage. We now model this notion via the UniqueExternalLinkage
linkage type. There are several changes here:
- Extended NamedDecl::getLinkage() to produce UniqueExternalLinkage
when the declaration is in an anonymous namespace.
- Added Type::getLinkage() to determine the linkage of a type, which
is defined as the minimum linkage of the types (when we're dealing
with a compound type that is not a struct/class/union).
- Extended NamedDecl::getLinkage() to consider the linkage of the
template arguments and template parameters of function template
specializations and class template specializations.
- Taught code generation to rely on NamedDecl::getLinkage() when
determining the linkage of variables and functions, also
considering the linkage of the types of those variables and
functions (C++ only). Map UniqueExternalLinkage to internal
linkage, taking out the explicit checks for
isInAnonymousNamespace().
This fixes much of PR5792, which, as discovered by Anders Carlsson, is
actually the reason behind the pass-manager assertion that causes the
majority of clang-on-clang regression test failures. With this fix,
Clang-built-Clang+LLVM passes 88% of its regression tests (up from
67%). The specific numbers are:
LLVM:
Expected Passes : 4006
Expected Failures : 32
Unsupported Tests : 40
Unexpected Failures: 736
Clang:
Expected Passes : 1903
Expected Failures : 14
Unexpected Failures: 75
Overall:
Expected Passes : 5909
Expected Failures : 46
Unsupported Tests : 40
Unexpected Failures: 811
Still to do:
- Improve testing
- Check whether we should allow the presence of types with
InternalLinkage (in addition to UniqueExternalLinkage) given
variables/functions internal linkage in C++, as mentioned in
PR5792.
- Determine how expensive the getLinkage() calls are in practice;
consider caching the result in NamedDecl.
- Assess the feasibility of Chris's idea in comment #1 of PR5792.
llvm-svn: 95216
2010-02-03 17:33:45 +08:00
|
|
|
if (Linkage L = PrevVar->getLinkage())
|
2009-11-26 06:24:25 +08:00
|
|
|
return L;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// C99 6.2.2p5:
|
|
|
|
// If the declaration of an identifier for an object has file
|
|
|
|
// scope and no storage-class specifier, its linkage is
|
|
|
|
// external.
|
When a function or variable somehow depends on a type or declaration
that is in an anonymous namespace, give that function or variable
internal linkage.
This change models an oddity of the C++ standard, where names declared
in an anonymous namespace have external linkage but, because anonymous
namespace are really "uniquely-named" namespaces, the names cannot be
referenced from other translation units. That means that they have
external linkage for semantic analysis, but the only sensible
implementation for code generation is to give them internal
linkage. We now model this notion via the UniqueExternalLinkage
linkage type. There are several changes here:
- Extended NamedDecl::getLinkage() to produce UniqueExternalLinkage
when the declaration is in an anonymous namespace.
- Added Type::getLinkage() to determine the linkage of a type, which
is defined as the minimum linkage of the types (when we're dealing
with a compound type that is not a struct/class/union).
- Extended NamedDecl::getLinkage() to consider the linkage of the
template arguments and template parameters of function template
specializations and class template specializations.
- Taught code generation to rely on NamedDecl::getLinkage() when
determining the linkage of variables and functions, also
considering the linkage of the types of those variables and
functions (C++ only). Map UniqueExternalLinkage to internal
linkage, taking out the explicit checks for
isInAnonymousNamespace().
This fixes much of PR5792, which, as discovered by Anders Carlsson, is
actually the reason behind the pass-manager assertion that causes the
majority of clang-on-clang regression test failures. With this fix,
Clang-built-Clang+LLVM passes 88% of its regression tests (up from
67%). The specific numbers are:
LLVM:
Expected Passes : 4006
Expected Failures : 32
Unsupported Tests : 40
Unexpected Failures: 736
Clang:
Expected Passes : 1903
Expected Failures : 14
Unexpected Failures: 75
Overall:
Expected Passes : 5909
Expected Failures : 46
Unsupported Tests : 40
Unexpected Failures: 811
Still to do:
- Improve testing
- Check whether we should allow the presence of types with
InternalLinkage (in addition to UniqueExternalLinkage) given
variables/functions internal linkage in C++, as mentioned in
PR5792.
- Determine how expensive the getLinkage() calls are in practice;
consider caching the result in NamedDecl.
- Assess the feasibility of Chris's idea in comment #1 of PR5792.
llvm-svn: 95216
2010-02-03 17:33:45 +08:00
|
|
|
if (Var->isInAnonymousNamespace())
|
|
|
|
return UniqueExternalLinkage;
|
|
|
|
|
|
|
|
return ExternalLinkage;
|
2009-11-26 06:24:25 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
// - a function, unless it has internal linkage; or
|
|
|
|
if (const FunctionDecl *Function = dyn_cast<FunctionDecl>(D)) {
|
|
|
|
// C99 6.2.2p5:
|
|
|
|
// If the declaration of an identifier for a function has no
|
|
|
|
// storage-class specifier, its linkage is determined exactly
|
|
|
|
// as if it were declared with the storage-class specifier
|
|
|
|
// extern.
|
|
|
|
if (!Context.getLangOptions().CPlusPlus &&
|
|
|
|
(Function->getStorageClass() == FunctionDecl::Extern ||
|
|
|
|
Function->getStorageClass() == FunctionDecl::PrivateExtern ||
|
|
|
|
Function->getStorageClass() == FunctionDecl::None)) {
|
|
|
|
// C99 6.2.2p4:
|
|
|
|
// For an identifier declared with the storage-class specifier
|
|
|
|
// extern in a scope in which a prior declaration of that
|
|
|
|
// identifier is visible, if the prior declaration specifies
|
|
|
|
// internal or external linkage, the linkage of the identifier
|
|
|
|
// at the later declaration is the same as the linkage
|
|
|
|
// specified at the prior declaration. If no prior declaration
|
|
|
|
// is visible, or if the prior declaration specifies no
|
|
|
|
// linkage, then the identifier has external linkage.
|
|
|
|
if (const FunctionDecl *PrevFunc = Function->getPreviousDeclaration()) {
|
When a function or variable somehow depends on a type or declaration
that is in an anonymous namespace, give that function or variable
internal linkage.
This change models an oddity of the C++ standard, where names declared
in an anonymous namespace have external linkage but, because anonymous
namespace are really "uniquely-named" namespaces, the names cannot be
referenced from other translation units. That means that they have
external linkage for semantic analysis, but the only sensible
implementation for code generation is to give them internal
linkage. We now model this notion via the UniqueExternalLinkage
linkage type. There are several changes here:
- Extended NamedDecl::getLinkage() to produce UniqueExternalLinkage
when the declaration is in an anonymous namespace.
- Added Type::getLinkage() to determine the linkage of a type, which
is defined as the minimum linkage of the types (when we're dealing
with a compound type that is not a struct/class/union).
- Extended NamedDecl::getLinkage() to consider the linkage of the
template arguments and template parameters of function template
specializations and class template specializations.
- Taught code generation to rely on NamedDecl::getLinkage() when
determining the linkage of variables and functions, also
considering the linkage of the types of those variables and
functions (C++ only). Map UniqueExternalLinkage to internal
linkage, taking out the explicit checks for
isInAnonymousNamespace().
This fixes much of PR5792, which, as discovered by Anders Carlsson, is
actually the reason behind the pass-manager assertion that causes the
majority of clang-on-clang regression test failures. With this fix,
Clang-built-Clang+LLVM passes 88% of its regression tests (up from
67%). The specific numbers are:
LLVM:
Expected Passes : 4006
Expected Failures : 32
Unsupported Tests : 40
Unexpected Failures: 736
Clang:
Expected Passes : 1903
Expected Failures : 14
Unexpected Failures: 75
Overall:
Expected Passes : 5909
Expected Failures : 46
Unsupported Tests : 40
Unexpected Failures: 811
Still to do:
- Improve testing
- Check whether we should allow the presence of types with
InternalLinkage (in addition to UniqueExternalLinkage) given
variables/functions internal linkage in C++, as mentioned in
PR5792.
- Determine how expensive the getLinkage() calls are in practice;
consider caching the result in NamedDecl.
- Assess the feasibility of Chris's idea in comment #1 of PR5792.
llvm-svn: 95216
2010-02-03 17:33:45 +08:00
|
|
|
if (Linkage L = PrevFunc->getLinkage())
|
2009-11-26 06:24:25 +08:00
|
|
|
return L;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
When a function or variable somehow depends on a type or declaration
that is in an anonymous namespace, give that function or variable
internal linkage.
This change models an oddity of the C++ standard, where names declared
in an anonymous namespace have external linkage but, because anonymous
namespace are really "uniquely-named" namespaces, the names cannot be
referenced from other translation units. That means that they have
external linkage for semantic analysis, but the only sensible
implementation for code generation is to give them internal
linkage. We now model this notion via the UniqueExternalLinkage
linkage type. There are several changes here:
- Extended NamedDecl::getLinkage() to produce UniqueExternalLinkage
when the declaration is in an anonymous namespace.
- Added Type::getLinkage() to determine the linkage of a type, which
is defined as the minimum linkage of the types (when we're dealing
with a compound type that is not a struct/class/union).
- Extended NamedDecl::getLinkage() to consider the linkage of the
template arguments and template parameters of function template
specializations and class template specializations.
- Taught code generation to rely on NamedDecl::getLinkage() when
determining the linkage of variables and functions, also
considering the linkage of the types of those variables and
functions (C++ only). Map UniqueExternalLinkage to internal
linkage, taking out the explicit checks for
isInAnonymousNamespace().
This fixes much of PR5792, which, as discovered by Anders Carlsson, is
actually the reason behind the pass-manager assertion that causes the
majority of clang-on-clang regression test failures. With this fix,
Clang-built-Clang+LLVM passes 88% of its regression tests (up from
67%). The specific numbers are:
LLVM:
Expected Passes : 4006
Expected Failures : 32
Unsupported Tests : 40
Unexpected Failures: 736
Clang:
Expected Passes : 1903
Expected Failures : 14
Unexpected Failures: 75
Overall:
Expected Passes : 5909
Expected Failures : 46
Unsupported Tests : 40
Unexpected Failures: 811
Still to do:
- Improve testing
- Check whether we should allow the presence of types with
InternalLinkage (in addition to UniqueExternalLinkage) given
variables/functions internal linkage in C++, as mentioned in
PR5792.
- Determine how expensive the getLinkage() calls are in practice;
consider caching the result in NamedDecl.
- Assess the feasibility of Chris's idea in comment #1 of PR5792.
llvm-svn: 95216
2010-02-03 17:33:45 +08:00
|
|
|
if (Function->isInAnonymousNamespace())
|
|
|
|
return UniqueExternalLinkage;
|
|
|
|
|
|
|
|
if (FunctionTemplateSpecializationInfo *SpecInfo
|
|
|
|
= Function->getTemplateSpecializationInfo()) {
|
|
|
|
Linkage L = SpecInfo->getTemplate()->getLinkage();
|
|
|
|
const TemplateArgumentList &TemplateArgs = *SpecInfo->TemplateArguments;
|
|
|
|
L = minLinkage(L,
|
|
|
|
getLinkageForTemplateArgumentList(
|
|
|
|
TemplateArgs.getFlatArgumentList(),
|
|
|
|
TemplateArgs.flat_size()));
|
|
|
|
return L;
|
|
|
|
}
|
|
|
|
|
|
|
|
return ExternalLinkage;
|
2009-11-26 06:24:25 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
// - a named class (Clause 9), or an unnamed class defined in a
|
|
|
|
// typedef declaration in which the class has the typedef name
|
|
|
|
// for linkage purposes (7.1.3); or
|
|
|
|
// - a named enumeration (7.2), or an unnamed enumeration
|
|
|
|
// defined in a typedef declaration in which the enumeration
|
|
|
|
// has the typedef name for linkage purposes (7.1.3); or
|
|
|
|
if (const TagDecl *Tag = dyn_cast<TagDecl>(D))
|
When a function or variable somehow depends on a type or declaration
that is in an anonymous namespace, give that function or variable
internal linkage.
This change models an oddity of the C++ standard, where names declared
in an anonymous namespace have external linkage but, because anonymous
namespace are really "uniquely-named" namespaces, the names cannot be
referenced from other translation units. That means that they have
external linkage for semantic analysis, but the only sensible
implementation for code generation is to give them internal
linkage. We now model this notion via the UniqueExternalLinkage
linkage type. There are several changes here:
- Extended NamedDecl::getLinkage() to produce UniqueExternalLinkage
when the declaration is in an anonymous namespace.
- Added Type::getLinkage() to determine the linkage of a type, which
is defined as the minimum linkage of the types (when we're dealing
with a compound type that is not a struct/class/union).
- Extended NamedDecl::getLinkage() to consider the linkage of the
template arguments and template parameters of function template
specializations and class template specializations.
- Taught code generation to rely on NamedDecl::getLinkage() when
determining the linkage of variables and functions, also
considering the linkage of the types of those variables and
functions (C++ only). Map UniqueExternalLinkage to internal
linkage, taking out the explicit checks for
isInAnonymousNamespace().
This fixes much of PR5792, which, as discovered by Anders Carlsson, is
actually the reason behind the pass-manager assertion that causes the
majority of clang-on-clang regression test failures. With this fix,
Clang-built-Clang+LLVM passes 88% of its regression tests (up from
67%). The specific numbers are:
LLVM:
Expected Passes : 4006
Expected Failures : 32
Unsupported Tests : 40
Unexpected Failures: 736
Clang:
Expected Passes : 1903
Expected Failures : 14
Unexpected Failures: 75
Overall:
Expected Passes : 5909
Expected Failures : 46
Unsupported Tests : 40
Unexpected Failures: 811
Still to do:
- Improve testing
- Check whether we should allow the presence of types with
InternalLinkage (in addition to UniqueExternalLinkage) given
variables/functions internal linkage in C++, as mentioned in
PR5792.
- Determine how expensive the getLinkage() calls are in practice;
consider caching the result in NamedDecl.
- Assess the feasibility of Chris's idea in comment #1 of PR5792.
llvm-svn: 95216
2010-02-03 17:33:45 +08:00
|
|
|
if (Tag->getDeclName() || Tag->getTypedefForAnonDecl()) {
|
|
|
|
if (Tag->isInAnonymousNamespace())
|
|
|
|
return UniqueExternalLinkage;
|
|
|
|
|
|
|
|
// If this is a class template specialization, consider the
|
|
|
|
// linkage of the template and template arguments.
|
|
|
|
if (const ClassTemplateSpecializationDecl *Spec
|
|
|
|
= dyn_cast<ClassTemplateSpecializationDecl>(Tag)) {
|
|
|
|
const TemplateArgumentList &TemplateArgs = Spec->getTemplateArgs();
|
|
|
|
Linkage L = getLinkageForTemplateArgumentList(
|
|
|
|
TemplateArgs.getFlatArgumentList(),
|
|
|
|
TemplateArgs.flat_size());
|
|
|
|
return minLinkage(L, Spec->getSpecializedTemplate()->getLinkage());
|
|
|
|
}
|
|
|
|
|
|
|
|
return ExternalLinkage;
|
|
|
|
}
|
2009-11-26 06:24:25 +08:00
|
|
|
|
|
|
|
// - an enumerator belonging to an enumeration with external linkage;
|
When a function or variable somehow depends on a type or declaration
that is in an anonymous namespace, give that function or variable
internal linkage.
This change models an oddity of the C++ standard, where names declared
in an anonymous namespace have external linkage but, because anonymous
namespace are really "uniquely-named" namespaces, the names cannot be
referenced from other translation units. That means that they have
external linkage for semantic analysis, but the only sensible
implementation for code generation is to give them internal
linkage. We now model this notion via the UniqueExternalLinkage
linkage type. There are several changes here:
- Extended NamedDecl::getLinkage() to produce UniqueExternalLinkage
when the declaration is in an anonymous namespace.
- Added Type::getLinkage() to determine the linkage of a type, which
is defined as the minimum linkage of the types (when we're dealing
with a compound type that is not a struct/class/union).
- Extended NamedDecl::getLinkage() to consider the linkage of the
template arguments and template parameters of function template
specializations and class template specializations.
- Taught code generation to rely on NamedDecl::getLinkage() when
determining the linkage of variables and functions, also
considering the linkage of the types of those variables and
functions (C++ only). Map UniqueExternalLinkage to internal
linkage, taking out the explicit checks for
isInAnonymousNamespace().
This fixes much of PR5792, which, as discovered by Anders Carlsson, is
actually the reason behind the pass-manager assertion that causes the
majority of clang-on-clang regression test failures. With this fix,
Clang-built-Clang+LLVM passes 88% of its regression tests (up from
67%). The specific numbers are:
LLVM:
Expected Passes : 4006
Expected Failures : 32
Unsupported Tests : 40
Unexpected Failures: 736
Clang:
Expected Passes : 1903
Expected Failures : 14
Unexpected Failures: 75
Overall:
Expected Passes : 5909
Expected Failures : 46
Unsupported Tests : 40
Unexpected Failures: 811
Still to do:
- Improve testing
- Check whether we should allow the presence of types with
InternalLinkage (in addition to UniqueExternalLinkage) given
variables/functions internal linkage in C++, as mentioned in
PR5792.
- Determine how expensive the getLinkage() calls are in practice;
consider caching the result in NamedDecl.
- Assess the feasibility of Chris's idea in comment #1 of PR5792.
llvm-svn: 95216
2010-02-03 17:33:45 +08:00
|
|
|
if (isa<EnumConstantDecl>(D)) {
|
|
|
|
Linkage L = cast<NamedDecl>(D->getDeclContext())->getLinkage();
|
|
|
|
if (isExternalLinkage(L))
|
|
|
|
return L;
|
|
|
|
}
|
2009-11-26 06:24:25 +08:00
|
|
|
|
|
|
|
// - a template, unless it is a function template that has
|
|
|
|
// internal linkage (Clause 14);
|
When a function or variable somehow depends on a type or declaration
that is in an anonymous namespace, give that function or variable
internal linkage.
This change models an oddity of the C++ standard, where names declared
in an anonymous namespace have external linkage but, because anonymous
namespace are really "uniquely-named" namespaces, the names cannot be
referenced from other translation units. That means that they have
external linkage for semantic analysis, but the only sensible
implementation for code generation is to give them internal
linkage. We now model this notion via the UniqueExternalLinkage
linkage type. There are several changes here:
- Extended NamedDecl::getLinkage() to produce UniqueExternalLinkage
when the declaration is in an anonymous namespace.
- Added Type::getLinkage() to determine the linkage of a type, which
is defined as the minimum linkage of the types (when we're dealing
with a compound type that is not a struct/class/union).
- Extended NamedDecl::getLinkage() to consider the linkage of the
template arguments and template parameters of function template
specializations and class template specializations.
- Taught code generation to rely on NamedDecl::getLinkage() when
determining the linkage of variables and functions, also
considering the linkage of the types of those variables and
functions (C++ only). Map UniqueExternalLinkage to internal
linkage, taking out the explicit checks for
isInAnonymousNamespace().
This fixes much of PR5792, which, as discovered by Anders Carlsson, is
actually the reason behind the pass-manager assertion that causes the
majority of clang-on-clang regression test failures. With this fix,
Clang-built-Clang+LLVM passes 88% of its regression tests (up from
67%). The specific numbers are:
LLVM:
Expected Passes : 4006
Expected Failures : 32
Unsupported Tests : 40
Unexpected Failures: 736
Clang:
Expected Passes : 1903
Expected Failures : 14
Unexpected Failures: 75
Overall:
Expected Passes : 5909
Expected Failures : 46
Unsupported Tests : 40
Unexpected Failures: 811
Still to do:
- Improve testing
- Check whether we should allow the presence of types with
InternalLinkage (in addition to UniqueExternalLinkage) given
variables/functions internal linkage in C++, as mentioned in
PR5792.
- Determine how expensive the getLinkage() calls are in practice;
consider caching the result in NamedDecl.
- Assess the feasibility of Chris's idea in comment #1 of PR5792.
llvm-svn: 95216
2010-02-03 17:33:45 +08:00
|
|
|
if (const TemplateDecl *Template = dyn_cast<TemplateDecl>(D)) {
|
|
|
|
if (D->isInAnonymousNamespace())
|
|
|
|
return UniqueExternalLinkage;
|
|
|
|
|
|
|
|
return getLinkageForTemplateParameterList(
|
|
|
|
Template->getTemplateParameters());
|
|
|
|
}
|
2009-11-26 06:24:25 +08:00
|
|
|
|
|
|
|
// - a namespace (7.3), unless it is declared within an unnamed
|
|
|
|
// namespace.
|
|
|
|
if (isa<NamespaceDecl>(D) && !D->isInAnonymousNamespace())
|
When a function or variable somehow depends on a type or declaration
that is in an anonymous namespace, give that function or variable
internal linkage.
This change models an oddity of the C++ standard, where names declared
in an anonymous namespace have external linkage but, because anonymous
namespace are really "uniquely-named" namespaces, the names cannot be
referenced from other translation units. That means that they have
external linkage for semantic analysis, but the only sensible
implementation for code generation is to give them internal
linkage. We now model this notion via the UniqueExternalLinkage
linkage type. There are several changes here:
- Extended NamedDecl::getLinkage() to produce UniqueExternalLinkage
when the declaration is in an anonymous namespace.
- Added Type::getLinkage() to determine the linkage of a type, which
is defined as the minimum linkage of the types (when we're dealing
with a compound type that is not a struct/class/union).
- Extended NamedDecl::getLinkage() to consider the linkage of the
template arguments and template parameters of function template
specializations and class template specializations.
- Taught code generation to rely on NamedDecl::getLinkage() when
determining the linkage of variables and functions, also
considering the linkage of the types of those variables and
functions (C++ only). Map UniqueExternalLinkage to internal
linkage, taking out the explicit checks for
isInAnonymousNamespace().
This fixes much of PR5792, which, as discovered by Anders Carlsson, is
actually the reason behind the pass-manager assertion that causes the
majority of clang-on-clang regression test failures. With this fix,
Clang-built-Clang+LLVM passes 88% of its regression tests (up from
67%). The specific numbers are:
LLVM:
Expected Passes : 4006
Expected Failures : 32
Unsupported Tests : 40
Unexpected Failures: 736
Clang:
Expected Passes : 1903
Expected Failures : 14
Unexpected Failures: 75
Overall:
Expected Passes : 5909
Expected Failures : 46
Unsupported Tests : 40
Unexpected Failures: 811
Still to do:
- Improve testing
- Check whether we should allow the presence of types with
InternalLinkage (in addition to UniqueExternalLinkage) given
variables/functions internal linkage in C++, as mentioned in
PR5792.
- Determine how expensive the getLinkage() calls are in practice;
consider caching the result in NamedDecl.
- Assess the feasibility of Chris's idea in comment #1 of PR5792.
llvm-svn: 95216
2010-02-03 17:33:45 +08:00
|
|
|
return ExternalLinkage;
|
2009-11-26 06:24:25 +08:00
|
|
|
|
When a function or variable somehow depends on a type or declaration
that is in an anonymous namespace, give that function or variable
internal linkage.
This change models an oddity of the C++ standard, where names declared
in an anonymous namespace have external linkage but, because anonymous
namespace are really "uniquely-named" namespaces, the names cannot be
referenced from other translation units. That means that they have
external linkage for semantic analysis, but the only sensible
implementation for code generation is to give them internal
linkage. We now model this notion via the UniqueExternalLinkage
linkage type. There are several changes here:
- Extended NamedDecl::getLinkage() to produce UniqueExternalLinkage
when the declaration is in an anonymous namespace.
- Added Type::getLinkage() to determine the linkage of a type, which
is defined as the minimum linkage of the types (when we're dealing
with a compound type that is not a struct/class/union).
- Extended NamedDecl::getLinkage() to consider the linkage of the
template arguments and template parameters of function template
specializations and class template specializations.
- Taught code generation to rely on NamedDecl::getLinkage() when
determining the linkage of variables and functions, also
considering the linkage of the types of those variables and
functions (C++ only). Map UniqueExternalLinkage to internal
linkage, taking out the explicit checks for
isInAnonymousNamespace().
This fixes much of PR5792, which, as discovered by Anders Carlsson, is
actually the reason behind the pass-manager assertion that causes the
majority of clang-on-clang regression test failures. With this fix,
Clang-built-Clang+LLVM passes 88% of its regression tests (up from
67%). The specific numbers are:
LLVM:
Expected Passes : 4006
Expected Failures : 32
Unsupported Tests : 40
Unexpected Failures: 736
Clang:
Expected Passes : 1903
Expected Failures : 14
Unexpected Failures: 75
Overall:
Expected Passes : 5909
Expected Failures : 46
Unsupported Tests : 40
Unexpected Failures: 811
Still to do:
- Improve testing
- Check whether we should allow the presence of types with
InternalLinkage (in addition to UniqueExternalLinkage) given
variables/functions internal linkage in C++, as mentioned in
PR5792.
- Determine how expensive the getLinkage() calls are in practice;
consider caching the result in NamedDecl.
- Assess the feasibility of Chris's idea in comment #1 of PR5792.
llvm-svn: 95216
2010-02-03 17:33:45 +08:00
|
|
|
return NoLinkage;
|
2009-11-26 06:24:25 +08:00
|
|
|
}
|
|
|
|
|
When a function or variable somehow depends on a type or declaration
that is in an anonymous namespace, give that function or variable
internal linkage.
This change models an oddity of the C++ standard, where names declared
in an anonymous namespace have external linkage but, because anonymous
namespace are really "uniquely-named" namespaces, the names cannot be
referenced from other translation units. That means that they have
external linkage for semantic analysis, but the only sensible
implementation for code generation is to give them internal
linkage. We now model this notion via the UniqueExternalLinkage
linkage type. There are several changes here:
- Extended NamedDecl::getLinkage() to produce UniqueExternalLinkage
when the declaration is in an anonymous namespace.
- Added Type::getLinkage() to determine the linkage of a type, which
is defined as the minimum linkage of the types (when we're dealing
with a compound type that is not a struct/class/union).
- Extended NamedDecl::getLinkage() to consider the linkage of the
template arguments and template parameters of function template
specializations and class template specializations.
- Taught code generation to rely on NamedDecl::getLinkage() when
determining the linkage of variables and functions, also
considering the linkage of the types of those variables and
functions (C++ only). Map UniqueExternalLinkage to internal
linkage, taking out the explicit checks for
isInAnonymousNamespace().
This fixes much of PR5792, which, as discovered by Anders Carlsson, is
actually the reason behind the pass-manager assertion that causes the
majority of clang-on-clang regression test failures. With this fix,
Clang-built-Clang+LLVM passes 88% of its regression tests (up from
67%). The specific numbers are:
LLVM:
Expected Passes : 4006
Expected Failures : 32
Unsupported Tests : 40
Unexpected Failures: 736
Clang:
Expected Passes : 1903
Expected Failures : 14
Unexpected Failures: 75
Overall:
Expected Passes : 5909
Expected Failures : 46
Unsupported Tests : 40
Unexpected Failures: 811
Still to do:
- Improve testing
- Check whether we should allow the presence of types with
InternalLinkage (in addition to UniqueExternalLinkage) given
variables/functions internal linkage in C++, as mentioned in
PR5792.
- Determine how expensive the getLinkage() calls are in practice;
consider caching the result in NamedDecl.
- Assess the feasibility of Chris's idea in comment #1 of PR5792.
llvm-svn: 95216
2010-02-03 17:33:45 +08:00
|
|
|
Linkage NamedDecl::getLinkage() const {
|
2009-11-26 06:24:25 +08:00
|
|
|
// Handle linkage for namespace-scope names.
|
|
|
|
if (getDeclContext()->getLookupContext()->isFileContext())
|
|
|
|
if (Linkage L = getLinkageForNamespaceScopeDecl(this))
|
|
|
|
return L;
|
|
|
|
|
|
|
|
// C++ [basic.link]p5:
|
|
|
|
// In addition, a member function, static data member, a named
|
|
|
|
// class or enumeration of class scope, or an unnamed class or
|
|
|
|
// enumeration defined in a class-scope typedef declaration such
|
|
|
|
// that the class or enumeration has the typedef name for linkage
|
|
|
|
// purposes (7.1.3), has external linkage if the name of the class
|
|
|
|
// has external linkage.
|
|
|
|
if (getDeclContext()->isRecord() &&
|
|
|
|
(isa<CXXMethodDecl>(this) || isa<VarDecl>(this) ||
|
|
|
|
(isa<TagDecl>(this) &&
|
When a function or variable somehow depends on a type or declaration
that is in an anonymous namespace, give that function or variable
internal linkage.
This change models an oddity of the C++ standard, where names declared
in an anonymous namespace have external linkage but, because anonymous
namespace are really "uniquely-named" namespaces, the names cannot be
referenced from other translation units. That means that they have
external linkage for semantic analysis, but the only sensible
implementation for code generation is to give them internal
linkage. We now model this notion via the UniqueExternalLinkage
linkage type. There are several changes here:
- Extended NamedDecl::getLinkage() to produce UniqueExternalLinkage
when the declaration is in an anonymous namespace.
- Added Type::getLinkage() to determine the linkage of a type, which
is defined as the minimum linkage of the types (when we're dealing
with a compound type that is not a struct/class/union).
- Extended NamedDecl::getLinkage() to consider the linkage of the
template arguments and template parameters of function template
specializations and class template specializations.
- Taught code generation to rely on NamedDecl::getLinkage() when
determining the linkage of variables and functions, also
considering the linkage of the types of those variables and
functions (C++ only). Map UniqueExternalLinkage to internal
linkage, taking out the explicit checks for
isInAnonymousNamespace().
This fixes much of PR5792, which, as discovered by Anders Carlsson, is
actually the reason behind the pass-manager assertion that causes the
majority of clang-on-clang regression test failures. With this fix,
Clang-built-Clang+LLVM passes 88% of its regression tests (up from
67%). The specific numbers are:
LLVM:
Expected Passes : 4006
Expected Failures : 32
Unsupported Tests : 40
Unexpected Failures: 736
Clang:
Expected Passes : 1903
Expected Failures : 14
Unexpected Failures: 75
Overall:
Expected Passes : 5909
Expected Failures : 46
Unsupported Tests : 40
Unexpected Failures: 811
Still to do:
- Improve testing
- Check whether we should allow the presence of types with
InternalLinkage (in addition to UniqueExternalLinkage) given
variables/functions internal linkage in C++, as mentioned in
PR5792.
- Determine how expensive the getLinkage() calls are in practice;
consider caching the result in NamedDecl.
- Assess the feasibility of Chris's idea in comment #1 of PR5792.
llvm-svn: 95216
2010-02-03 17:33:45 +08:00
|
|
|
(getDeclName() || cast<TagDecl>(this)->getTypedefForAnonDecl())))) {
|
|
|
|
Linkage L = cast<RecordDecl>(getDeclContext())->getLinkage();
|
|
|
|
if (isExternalLinkage(L))
|
|
|
|
return L;
|
|
|
|
}
|
2009-11-26 06:24:25 +08:00
|
|
|
|
|
|
|
// C++ [basic.link]p6:
|
|
|
|
// The name of a function declared in block scope and the name of
|
|
|
|
// an object declared by a block scope extern declaration have
|
|
|
|
// linkage. If there is a visible declaration of an entity with
|
|
|
|
// linkage having the same name and type, ignoring entities
|
|
|
|
// declared outside the innermost enclosing namespace scope, the
|
|
|
|
// block scope declaration declares that same entity and receives
|
|
|
|
// the linkage of the previous declaration. If there is more than
|
|
|
|
// one such matching entity, the program is ill-formed. Otherwise,
|
|
|
|
// if no matching entity is found, the block scope entity receives
|
|
|
|
// external linkage.
|
|
|
|
if (getLexicalDeclContext()->isFunctionOrMethod()) {
|
|
|
|
if (const FunctionDecl *Function = dyn_cast<FunctionDecl>(this)) {
|
|
|
|
if (Function->getPreviousDeclaration())
|
|
|
|
if (Linkage L = Function->getPreviousDeclaration()->getLinkage())
|
|
|
|
return L;
|
|
|
|
|
When a function or variable somehow depends on a type or declaration
that is in an anonymous namespace, give that function or variable
internal linkage.
This change models an oddity of the C++ standard, where names declared
in an anonymous namespace have external linkage but, because anonymous
namespace are really "uniquely-named" namespaces, the names cannot be
referenced from other translation units. That means that they have
external linkage for semantic analysis, but the only sensible
implementation for code generation is to give them internal
linkage. We now model this notion via the UniqueExternalLinkage
linkage type. There are several changes here:
- Extended NamedDecl::getLinkage() to produce UniqueExternalLinkage
when the declaration is in an anonymous namespace.
- Added Type::getLinkage() to determine the linkage of a type, which
is defined as the minimum linkage of the types (when we're dealing
with a compound type that is not a struct/class/union).
- Extended NamedDecl::getLinkage() to consider the linkage of the
template arguments and template parameters of function template
specializations and class template specializations.
- Taught code generation to rely on NamedDecl::getLinkage() when
determining the linkage of variables and functions, also
considering the linkage of the types of those variables and
functions (C++ only). Map UniqueExternalLinkage to internal
linkage, taking out the explicit checks for
isInAnonymousNamespace().
This fixes much of PR5792, which, as discovered by Anders Carlsson, is
actually the reason behind the pass-manager assertion that causes the
majority of clang-on-clang regression test failures. With this fix,
Clang-built-Clang+LLVM passes 88% of its regression tests (up from
67%). The specific numbers are:
LLVM:
Expected Passes : 4006
Expected Failures : 32
Unsupported Tests : 40
Unexpected Failures: 736
Clang:
Expected Passes : 1903
Expected Failures : 14
Unexpected Failures: 75
Overall:
Expected Passes : 5909
Expected Failures : 46
Unsupported Tests : 40
Unexpected Failures: 811
Still to do:
- Improve testing
- Check whether we should allow the presence of types with
InternalLinkage (in addition to UniqueExternalLinkage) given
variables/functions internal linkage in C++, as mentioned in
PR5792.
- Determine how expensive the getLinkage() calls are in practice;
consider caching the result in NamedDecl.
- Assess the feasibility of Chris's idea in comment #1 of PR5792.
llvm-svn: 95216
2010-02-03 17:33:45 +08:00
|
|
|
if (Function->isInAnonymousNamespace())
|
|
|
|
return UniqueExternalLinkage;
|
|
|
|
|
2009-11-26 06:24:25 +08:00
|
|
|
return ExternalLinkage;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (const VarDecl *Var = dyn_cast<VarDecl>(this))
|
|
|
|
if (Var->getStorageClass() == VarDecl::Extern ||
|
|
|
|
Var->getStorageClass() == VarDecl::PrivateExtern) {
|
|
|
|
if (Var->getPreviousDeclaration())
|
|
|
|
if (Linkage L = Var->getPreviousDeclaration()->getLinkage())
|
|
|
|
return L;
|
|
|
|
|
When a function or variable somehow depends on a type or declaration
that is in an anonymous namespace, give that function or variable
internal linkage.
This change models an oddity of the C++ standard, where names declared
in an anonymous namespace have external linkage but, because anonymous
namespace are really "uniquely-named" namespaces, the names cannot be
referenced from other translation units. That means that they have
external linkage for semantic analysis, but the only sensible
implementation for code generation is to give them internal
linkage. We now model this notion via the UniqueExternalLinkage
linkage type. There are several changes here:
- Extended NamedDecl::getLinkage() to produce UniqueExternalLinkage
when the declaration is in an anonymous namespace.
- Added Type::getLinkage() to determine the linkage of a type, which
is defined as the minimum linkage of the types (when we're dealing
with a compound type that is not a struct/class/union).
- Extended NamedDecl::getLinkage() to consider the linkage of the
template arguments and template parameters of function template
specializations and class template specializations.
- Taught code generation to rely on NamedDecl::getLinkage() when
determining the linkage of variables and functions, also
considering the linkage of the types of those variables and
functions (C++ only). Map UniqueExternalLinkage to internal
linkage, taking out the explicit checks for
isInAnonymousNamespace().
This fixes much of PR5792, which, as discovered by Anders Carlsson, is
actually the reason behind the pass-manager assertion that causes the
majority of clang-on-clang regression test failures. With this fix,
Clang-built-Clang+LLVM passes 88% of its regression tests (up from
67%). The specific numbers are:
LLVM:
Expected Passes : 4006
Expected Failures : 32
Unsupported Tests : 40
Unexpected Failures: 736
Clang:
Expected Passes : 1903
Expected Failures : 14
Unexpected Failures: 75
Overall:
Expected Passes : 5909
Expected Failures : 46
Unsupported Tests : 40
Unexpected Failures: 811
Still to do:
- Improve testing
- Check whether we should allow the presence of types with
InternalLinkage (in addition to UniqueExternalLinkage) given
variables/functions internal linkage in C++, as mentioned in
PR5792.
- Determine how expensive the getLinkage() calls are in practice;
consider caching the result in NamedDecl.
- Assess the feasibility of Chris's idea in comment #1 of PR5792.
llvm-svn: 95216
2010-02-03 17:33:45 +08:00
|
|
|
if (Var->isInAnonymousNamespace())
|
|
|
|
return UniqueExternalLinkage;
|
|
|
|
|
2009-11-26 06:24:25 +08:00
|
|
|
return ExternalLinkage;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// C++ [basic.link]p6:
|
|
|
|
// Names not covered by these rules have no linkage.
|
|
|
|
return NoLinkage;
|
When a function or variable somehow depends on a type or declaration
that is in an anonymous namespace, give that function or variable
internal linkage.
This change models an oddity of the C++ standard, where names declared
in an anonymous namespace have external linkage but, because anonymous
namespace are really "uniquely-named" namespaces, the names cannot be
referenced from other translation units. That means that they have
external linkage for semantic analysis, but the only sensible
implementation for code generation is to give them internal
linkage. We now model this notion via the UniqueExternalLinkage
linkage type. There are several changes here:
- Extended NamedDecl::getLinkage() to produce UniqueExternalLinkage
when the declaration is in an anonymous namespace.
- Added Type::getLinkage() to determine the linkage of a type, which
is defined as the minimum linkage of the types (when we're dealing
with a compound type that is not a struct/class/union).
- Extended NamedDecl::getLinkage() to consider the linkage of the
template arguments and template parameters of function template
specializations and class template specializations.
- Taught code generation to rely on NamedDecl::getLinkage() when
determining the linkage of variables and functions, also
considering the linkage of the types of those variables and
functions (C++ only). Map UniqueExternalLinkage to internal
linkage, taking out the explicit checks for
isInAnonymousNamespace().
This fixes much of PR5792, which, as discovered by Anders Carlsson, is
actually the reason behind the pass-manager assertion that causes the
majority of clang-on-clang regression test failures. With this fix,
Clang-built-Clang+LLVM passes 88% of its regression tests (up from
67%). The specific numbers are:
LLVM:
Expected Passes : 4006
Expected Failures : 32
Unsupported Tests : 40
Unexpected Failures: 736
Clang:
Expected Passes : 1903
Expected Failures : 14
Unexpected Failures: 75
Overall:
Expected Passes : 5909
Expected Failures : 46
Unsupported Tests : 40
Unexpected Failures: 811
Still to do:
- Improve testing
- Check whether we should allow the presence of types with
InternalLinkage (in addition to UniqueExternalLinkage) given
variables/functions internal linkage in C++, as mentioned in
PR5792.
- Determine how expensive the getLinkage() calls are in practice;
consider caching the result in NamedDecl.
- Assess the feasibility of Chris's idea in comment #1 of PR5792.
llvm-svn: 95216
2010-02-03 17:33:45 +08:00
|
|
|
}
|
2009-11-26 06:24:25 +08:00
|
|
|
|
2009-02-05 01:27:36 +08:00
|
|
|
std::string NamedDecl::getQualifiedNameAsString() const {
|
2009-09-09 02:24:21 +08:00
|
|
|
return getQualifiedNameAsString(getASTContext().getLangOptions());
|
|
|
|
}
|
|
|
|
|
|
|
|
std::string NamedDecl::getQualifiedNameAsString(const PrintingPolicy &P) const {
|
2009-10-19 04:26:12 +08:00
|
|
|
// FIXME: Collect contexts, then accumulate names to avoid unnecessary
|
|
|
|
// std::string thrashing.
|
2009-02-05 01:27:36 +08:00
|
|
|
std::vector<std::string> Names;
|
|
|
|
std::string QualName;
|
|
|
|
const DeclContext *Ctx = getDeclContext();
|
|
|
|
|
|
|
|
if (Ctx->isFunctionOrMethod())
|
|
|
|
return getNameAsString();
|
|
|
|
|
|
|
|
while (Ctx) {
|
2009-09-09 23:08:12 +08:00
|
|
|
if (const ClassTemplateSpecializationDecl *Spec
|
2009-05-19 01:01:57 +08:00
|
|
|
= dyn_cast<ClassTemplateSpecializationDecl>(Ctx)) {
|
|
|
|
const TemplateArgumentList &TemplateArgs = Spec->getTemplateArgs();
|
|
|
|
std::string TemplateArgsStr
|
|
|
|
= TemplateSpecializationType::PrintTemplateArgumentList(
|
|
|
|
TemplateArgs.getFlatArgumentList(),
|
2009-05-30 04:38:28 +08:00
|
|
|
TemplateArgs.flat_size(),
|
2009-09-09 02:24:21 +08:00
|
|
|
P);
|
2009-10-19 04:26:12 +08:00
|
|
|
Names.push_back(Spec->getIdentifier()->getNameStart() + TemplateArgsStr);
|
2009-12-25 07:15:03 +08:00
|
|
|
} else if (const NamespaceDecl *ND = dyn_cast<NamespaceDecl>(Ctx)) {
|
|
|
|
if (ND->isAnonymousNamespace())
|
|
|
|
Names.push_back("<anonymous namespace>");
|
|
|
|
else
|
|
|
|
Names.push_back(ND->getNameAsString());
|
|
|
|
} else if (const RecordDecl *RD = dyn_cast<RecordDecl>(Ctx)) {
|
|
|
|
if (!RD->getIdentifier()) {
|
|
|
|
std::string RecordString = "<anonymous ";
|
|
|
|
RecordString += RD->getKindName();
|
|
|
|
RecordString += ">";
|
|
|
|
Names.push_back(RecordString);
|
|
|
|
} else {
|
|
|
|
Names.push_back(RD->getNameAsString());
|
|
|
|
}
|
2009-12-28 11:19:38 +08:00
|
|
|
} else if (const FunctionDecl *FD = dyn_cast<FunctionDecl>(Ctx)) {
|
|
|
|
std::string Proto = FD->getNameAsString();
|
|
|
|
|
|
|
|
const FunctionProtoType *FT = 0;
|
|
|
|
if (FD->hasWrittenPrototype())
|
|
|
|
FT = dyn_cast<FunctionProtoType>(FD->getType()->getAs<FunctionType>());
|
|
|
|
|
|
|
|
Proto += "(";
|
|
|
|
if (FT) {
|
|
|
|
llvm::raw_string_ostream POut(Proto);
|
|
|
|
unsigned NumParams = FD->getNumParams();
|
|
|
|
for (unsigned i = 0; i < NumParams; ++i) {
|
|
|
|
if (i)
|
|
|
|
POut << ", ";
|
|
|
|
std::string Param;
|
|
|
|
FD->getParamDecl(i)->getType().getAsStringInternal(Param, P);
|
|
|
|
POut << Param;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (FT->isVariadic()) {
|
|
|
|
if (NumParams > 0)
|
|
|
|
POut << ", ";
|
|
|
|
POut << "...";
|
|
|
|
}
|
|
|
|
}
|
|
|
|
Proto += ")";
|
|
|
|
|
|
|
|
Names.push_back(Proto);
|
2009-05-19 01:01:57 +08:00
|
|
|
} else if (const NamedDecl *ND = dyn_cast<NamedDecl>(Ctx))
|
2009-02-05 01:27:36 +08:00
|
|
|
Names.push_back(ND->getNameAsString());
|
|
|
|
else
|
|
|
|
break;
|
|
|
|
|
|
|
|
Ctx = Ctx->getParent();
|
|
|
|
}
|
|
|
|
|
|
|
|
std::vector<std::string>::reverse_iterator
|
|
|
|
I = Names.rbegin(),
|
|
|
|
End = Names.rend();
|
|
|
|
|
|
|
|
for (; I!=End; ++I)
|
|
|
|
QualName += *I + "::";
|
|
|
|
|
2010-03-17 05:48:18 +08:00
|
|
|
if (getDeclName())
|
|
|
|
QualName += getNameAsString();
|
|
|
|
else
|
|
|
|
QualName += "<anonymous>";
|
2009-02-05 01:27:36 +08:00
|
|
|
|
|
|
|
return QualName;
|
|
|
|
}
|
|
|
|
|
2009-01-20 09:17:11 +08:00
|
|
|
bool NamedDecl::declarationReplaces(NamedDecl *OldD) const {
|
2008-12-24 05:05:05 +08:00
|
|
|
assert(getDeclName() == OldD->getDeclName() && "Declaration name mismatch");
|
|
|
|
|
2009-02-04 03:21:40 +08:00
|
|
|
// UsingDirectiveDecl's are not really NamedDecl's, and all have same name.
|
|
|
|
// We want to keep it, unless it nominates same namespace.
|
|
|
|
if (getKind() == Decl::UsingDirective) {
|
|
|
|
return cast<UsingDirectiveDecl>(this)->getNominatedNamespace() ==
|
|
|
|
cast<UsingDirectiveDecl>(OldD)->getNominatedNamespace();
|
|
|
|
}
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2008-12-24 05:05:05 +08:00
|
|
|
if (const FunctionDecl *FD = dyn_cast<FunctionDecl>(this))
|
|
|
|
// For function declarations, we keep track of redeclarations.
|
|
|
|
return FD->getPreviousDeclaration() == OldD;
|
|
|
|
|
2009-06-26 06:08:12 +08:00
|
|
|
// For function templates, the underlying function declarations are linked.
|
|
|
|
if (const FunctionTemplateDecl *FunctionTemplate
|
|
|
|
= dyn_cast<FunctionTemplateDecl>(this))
|
|
|
|
if (const FunctionTemplateDecl *OldFunctionTemplate
|
|
|
|
= dyn_cast<FunctionTemplateDecl>(OldD))
|
|
|
|
return FunctionTemplate->getTemplatedDecl()
|
|
|
|
->declarationReplaces(OldFunctionTemplate->getTemplatedDecl());
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-02-23 03:35:57 +08:00
|
|
|
// For method declarations, we keep track of redeclarations.
|
|
|
|
if (isa<ObjCMethodDecl>(this))
|
|
|
|
return false;
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-10-10 05:13:30 +08:00
|
|
|
if (isa<ObjCInterfaceDecl>(this) && isa<ObjCCompatibleAliasDecl>(OldD))
|
|
|
|
return true;
|
|
|
|
|
2009-11-17 13:59:44 +08:00
|
|
|
if (isa<UsingShadowDecl>(this) && isa<UsingShadowDecl>(OldD))
|
|
|
|
return cast<UsingShadowDecl>(this)->getTargetDecl() ==
|
|
|
|
cast<UsingShadowDecl>(OldD)->getTargetDecl();
|
|
|
|
|
2008-12-24 05:05:05 +08:00
|
|
|
// For non-function declarations, if the declarations are of the
|
|
|
|
// same kind then this must be a redeclaration, or semantic analysis
|
|
|
|
// would not have given us the new declaration.
|
|
|
|
return this->getKind() == OldD->getKind();
|
|
|
|
}
|
|
|
|
|
2009-02-25 04:03:32 +08:00
|
|
|
bool NamedDecl::hasLinkage() const {
|
2009-11-26 06:24:25 +08:00
|
|
|
return getLinkage() != NoLinkage;
|
2009-02-25 04:03:32 +08:00
|
|
|
}
|
2009-01-20 09:17:11 +08:00
|
|
|
|
2009-06-26 14:29:23 +08:00
|
|
|
NamedDecl *NamedDecl::getUnderlyingDecl() {
|
|
|
|
NamedDecl *ND = this;
|
|
|
|
while (true) {
|
2009-11-17 13:59:44 +08:00
|
|
|
if (UsingShadowDecl *UD = dyn_cast<UsingShadowDecl>(ND))
|
2009-06-26 14:29:23 +08:00
|
|
|
ND = UD->getTargetDecl();
|
|
|
|
else if (ObjCCompatibleAliasDecl *AD
|
|
|
|
= dyn_cast<ObjCCompatibleAliasDecl>(ND))
|
|
|
|
return AD->getClassInterface();
|
|
|
|
else
|
|
|
|
return ND;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2009-08-21 08:31:54 +08:00
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
// DeclaratorDecl Implementation
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
|
2010-03-15 18:12:16 +08:00
|
|
|
DeclaratorDecl::~DeclaratorDecl() {}
|
|
|
|
void DeclaratorDecl::Destroy(ASTContext &C) {
|
|
|
|
if (hasExtInfo())
|
|
|
|
C.Deallocate(getExtInfo());
|
|
|
|
ValueDecl::Destroy(C);
|
|
|
|
}
|
|
|
|
|
2009-08-21 08:31:54 +08:00
|
|
|
SourceLocation DeclaratorDecl::getTypeSpecStartLoc() const {
|
2009-10-18 09:05:36 +08:00
|
|
|
if (DeclInfo) {
|
2010-03-15 18:12:16 +08:00
|
|
|
TypeLoc TL = getTypeSourceInfo()->getTypeLoc();
|
2009-10-18 09:05:36 +08:00
|
|
|
while (true) {
|
|
|
|
TypeLoc NextTL = TL.getNextTypeLoc();
|
|
|
|
if (!NextTL)
|
|
|
|
return TL.getSourceRange().getBegin();
|
|
|
|
TL = NextTL;
|
|
|
|
}
|
|
|
|
}
|
2009-08-21 08:31:54 +08:00
|
|
|
return SourceLocation();
|
|
|
|
}
|
|
|
|
|
2010-03-15 18:12:16 +08:00
|
|
|
void DeclaratorDecl::setQualifierInfo(NestedNameSpecifier *Qualifier,
|
|
|
|
SourceRange QualifierRange) {
|
|
|
|
if (Qualifier) {
|
|
|
|
// Make sure the extended decl info is allocated.
|
|
|
|
if (!hasExtInfo()) {
|
|
|
|
// Save (non-extended) type source info pointer.
|
|
|
|
TypeSourceInfo *savedTInfo = DeclInfo.get<TypeSourceInfo*>();
|
|
|
|
// Allocate external info struct.
|
|
|
|
DeclInfo = new (getASTContext()) ExtInfo;
|
|
|
|
// Restore savedTInfo into (extended) decl info.
|
|
|
|
getExtInfo()->TInfo = savedTInfo;
|
|
|
|
}
|
|
|
|
// Set qualifier info.
|
|
|
|
getExtInfo()->NNS = Qualifier;
|
|
|
|
getExtInfo()->NNSRange = QualifierRange;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
// Here Qualifier == 0, i.e., we are removing the qualifier (if any).
|
|
|
|
assert(QualifierRange.isInvalid());
|
|
|
|
if (hasExtInfo()) {
|
|
|
|
// Save type source info pointer.
|
|
|
|
TypeSourceInfo *savedTInfo = getExtInfo()->TInfo;
|
|
|
|
// Deallocate the extended decl info.
|
|
|
|
getASTContext().Deallocate(getExtInfo());
|
|
|
|
// Restore savedTInfo into (non-extended) decl info.
|
|
|
|
DeclInfo = savedTInfo;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2008-12-18 07:39:55 +08:00
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
// VarDecl Implementation
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
|
2010-01-27 06:01:41 +08:00
|
|
|
const char *VarDecl::getStorageClassSpecifierString(StorageClass SC) {
|
|
|
|
switch (SC) {
|
|
|
|
case VarDecl::None: break;
|
|
|
|
case VarDecl::Auto: return "auto"; break;
|
|
|
|
case VarDecl::Extern: return "extern"; break;
|
|
|
|
case VarDecl::PrivateExtern: return "__private_extern__"; break;
|
|
|
|
case VarDecl::Register: return "register"; break;
|
|
|
|
case VarDecl::Static: return "static"; break;
|
|
|
|
}
|
|
|
|
|
|
|
|
assert(0 && "Invalid storage class");
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2009-01-20 09:17:11 +08:00
|
|
|
VarDecl *VarDecl::Create(ASTContext &C, DeclContext *DC, SourceLocation L,
|
2009-12-07 10:54:59 +08:00
|
|
|
IdentifierInfo *Id, QualType T, TypeSourceInfo *TInfo,
|
2009-08-21 08:31:54 +08:00
|
|
|
StorageClass S) {
|
2009-12-07 10:54:59 +08:00
|
|
|
return new (C) VarDecl(Var, DC, L, Id, T, TInfo, S);
|
2008-12-18 07:39:55 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
void VarDecl::Destroy(ASTContext& C) {
|
2009-02-05 23:12:41 +08:00
|
|
|
Expr *Init = getInit();
|
2009-05-27 02:54:04 +08:00
|
|
|
if (Init) {
|
2009-02-05 23:12:41 +08:00
|
|
|
Init->Destroy(C);
|
2009-05-27 02:54:04 +08:00
|
|
|
if (EvaluatedStmt *Eval = this->Init.dyn_cast<EvaluatedStmt *>()) {
|
|
|
|
Eval->~EvaluatedStmt();
|
|
|
|
C.Deallocate(Eval);
|
|
|
|
}
|
|
|
|
}
|
2008-12-18 07:39:55 +08:00
|
|
|
this->~VarDecl();
|
2010-03-15 18:12:16 +08:00
|
|
|
DeclaratorDecl::Destroy(C);
|
2008-12-18 07:39:55 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
VarDecl::~VarDecl() {
|
|
|
|
}
|
|
|
|
|
2009-06-20 16:09:14 +08:00
|
|
|
SourceRange VarDecl::getSourceRange() const {
|
2010-01-23 03:49:59 +08:00
|
|
|
SourceLocation Start = getTypeSpecStartLoc();
|
|
|
|
if (Start.isInvalid())
|
|
|
|
Start = getLocation();
|
|
|
|
|
2009-06-20 16:09:14 +08:00
|
|
|
if (getInit())
|
2010-01-23 03:49:59 +08:00
|
|
|
return SourceRange(Start, getInit()->getLocEnd());
|
|
|
|
return SourceRange(Start, getLocation());
|
2009-06-20 16:09:14 +08:00
|
|
|
}
|
|
|
|
|
2010-01-27 06:01:41 +08:00
|
|
|
bool VarDecl::isExternC() const {
|
|
|
|
ASTContext &Context = getASTContext();
|
|
|
|
if (!Context.getLangOptions().CPlusPlus)
|
|
|
|
return (getDeclContext()->isTranslationUnit() &&
|
|
|
|
getStorageClass() != Static) ||
|
|
|
|
(getDeclContext()->isFunctionOrMethod() && hasExternalStorage());
|
|
|
|
|
|
|
|
for (const DeclContext *DC = getDeclContext(); !DC->isTranslationUnit();
|
|
|
|
DC = DC->getParent()) {
|
|
|
|
if (const LinkageSpecDecl *Linkage = dyn_cast<LinkageSpecDecl>(DC)) {
|
|
|
|
if (Linkage->getLanguage() == LinkageSpecDecl::lang_c)
|
|
|
|
return getStorageClass() != Static;
|
|
|
|
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (DC->isFunctionOrMethod())
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
VarDecl *VarDecl::getCanonicalDecl() {
|
|
|
|
return getFirstDeclaration();
|
|
|
|
}
|
|
|
|
|
2010-02-01 06:27:38 +08:00
|
|
|
VarDecl::DefinitionKind VarDecl::isThisDeclarationADefinition() const {
|
|
|
|
// C++ [basic.def]p2:
|
|
|
|
// A declaration is a definition unless [...] it contains the 'extern'
|
|
|
|
// specifier or a linkage-specification and neither an initializer [...],
|
|
|
|
// it declares a static data member in a class declaration [...].
|
|
|
|
// C++ [temp.expl.spec]p15:
|
|
|
|
// An explicit specialization of a static data member of a template is a
|
|
|
|
// definition if the declaration includes an initializer; otherwise, it is
|
|
|
|
// a declaration.
|
|
|
|
if (isStaticDataMember()) {
|
|
|
|
if (isOutOfLine() && (hasInit() ||
|
|
|
|
getTemplateSpecializationKind() != TSK_ExplicitSpecialization))
|
|
|
|
return Definition;
|
|
|
|
else
|
|
|
|
return DeclarationOnly;
|
|
|
|
}
|
|
|
|
// C99 6.7p5:
|
|
|
|
// A definition of an identifier is a declaration for that identifier that
|
|
|
|
// [...] causes storage to be reserved for that object.
|
|
|
|
// Note: that applies for all non-file-scope objects.
|
|
|
|
// C99 6.9.2p1:
|
|
|
|
// If the declaration of an identifier for an object has file scope and an
|
|
|
|
// initializer, the declaration is an external definition for the identifier
|
|
|
|
if (hasInit())
|
|
|
|
return Definition;
|
|
|
|
// AST for 'extern "C" int foo;' is annotated with 'extern'.
|
|
|
|
if (hasExternalStorage())
|
|
|
|
return DeclarationOnly;
|
|
|
|
|
|
|
|
// C99 6.9.2p2:
|
|
|
|
// A declaration of an object that has file scope without an initializer,
|
|
|
|
// and without a storage class specifier or the scs 'static', constitutes
|
|
|
|
// a tentative definition.
|
|
|
|
// No such thing in C++.
|
|
|
|
if (!getASTContext().getLangOptions().CPlusPlus && isFileVarDecl())
|
|
|
|
return TentativeDefinition;
|
|
|
|
|
|
|
|
// What's left is (in C, block-scope) declarations without initializers or
|
|
|
|
// external storage. These are definitions.
|
|
|
|
return Definition;
|
|
|
|
}
|
|
|
|
|
|
|
|
VarDecl *VarDecl::getActingDefinition() {
|
|
|
|
DefinitionKind Kind = isThisDeclarationADefinition();
|
|
|
|
if (Kind != TentativeDefinition)
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
VarDecl *LastTentative = false;
|
|
|
|
VarDecl *First = getFirstDeclaration();
|
|
|
|
for (redecl_iterator I = First->redecls_begin(), E = First->redecls_end();
|
|
|
|
I != E; ++I) {
|
|
|
|
Kind = (*I)->isThisDeclarationADefinition();
|
|
|
|
if (Kind == Definition)
|
|
|
|
return 0;
|
|
|
|
else if (Kind == TentativeDefinition)
|
|
|
|
LastTentative = *I;
|
|
|
|
}
|
|
|
|
return LastTentative;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool VarDecl::isTentativeDefinitionNow() const {
|
|
|
|
DefinitionKind Kind = isThisDeclarationADefinition();
|
|
|
|
if (Kind != TentativeDefinition)
|
|
|
|
return false;
|
|
|
|
|
|
|
|
for (redecl_iterator I = redecls_begin(), E = redecls_end(); I != E; ++I) {
|
|
|
|
if ((*I)->isThisDeclarationADefinition() == Definition)
|
|
|
|
return false;
|
|
|
|
}
|
2010-02-02 04:16:42 +08:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
VarDecl *VarDecl::getDefinition() {
|
2010-02-03 01:55:12 +08:00
|
|
|
VarDecl *First = getFirstDeclaration();
|
|
|
|
for (redecl_iterator I = First->redecls_begin(), E = First->redecls_end();
|
|
|
|
I != E; ++I) {
|
2010-02-02 04:16:42 +08:00
|
|
|
if ((*I)->isThisDeclarationADefinition() == Definition)
|
|
|
|
return *I;
|
|
|
|
}
|
|
|
|
return 0;
|
2010-02-01 06:27:38 +08:00
|
|
|
}
|
|
|
|
|
2010-02-02 04:16:42 +08:00
|
|
|
const Expr *VarDecl::getAnyInitializer(const VarDecl *&D) const {
|
2010-01-27 06:01:41 +08:00
|
|
|
redecl_iterator I = redecls_begin(), E = redecls_end();
|
|
|
|
while (I != E && !I->getInit())
|
|
|
|
++I;
|
|
|
|
|
|
|
|
if (I != E) {
|
2010-02-02 04:16:42 +08:00
|
|
|
D = *I;
|
2010-01-27 06:01:41 +08:00
|
|
|
return I->getInit();
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2009-10-15 05:29:40 +08:00
|
|
|
bool VarDecl::isOutOfLine() const {
|
|
|
|
if (Decl::isOutOfLine())
|
|
|
|
return true;
|
2010-02-21 15:08:09 +08:00
|
|
|
|
|
|
|
if (!isStaticDataMember())
|
|
|
|
return false;
|
|
|
|
|
2009-10-15 05:29:40 +08:00
|
|
|
// If this static data member was instantiated from a static data member of
|
|
|
|
// a class template, check whether that static data member was defined
|
|
|
|
// out-of-line.
|
|
|
|
if (VarDecl *VD = getInstantiatedFromStaticDataMember())
|
|
|
|
return VD->isOutOfLine();
|
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2009-10-28 02:42:08 +08:00
|
|
|
VarDecl *VarDecl::getOutOfLineDefinition() {
|
|
|
|
if (!isStaticDataMember())
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
for (VarDecl::redecl_iterator RD = redecls_begin(), RDEnd = redecls_end();
|
|
|
|
RD != RDEnd; ++RD) {
|
|
|
|
if (RD->getLexicalDeclContext()->isFileContext())
|
|
|
|
return *RD;
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2010-02-11 09:19:42 +08:00
|
|
|
void VarDecl::setInit(Expr *I) {
|
2010-01-27 06:01:41 +08:00
|
|
|
if (EvaluatedStmt *Eval = Init.dyn_cast<EvaluatedStmt *>()) {
|
|
|
|
Eval->~EvaluatedStmt();
|
2010-02-11 09:19:42 +08:00
|
|
|
getASTContext().Deallocate(Eval);
|
2010-01-27 06:01:41 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
Init = I;
|
|
|
|
}
|
|
|
|
|
2009-10-15 05:29:40 +08:00
|
|
|
VarDecl *VarDecl::getInstantiatedFromStaticDataMember() const {
|
2009-10-13 04:18:28 +08:00
|
|
|
if (MemberSpecializationInfo *MSI = getMemberSpecializationInfo())
|
2009-10-08 15:24:58 +08:00
|
|
|
return cast<VarDecl>(MSI->getInstantiatedFrom());
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2009-10-15 04:14:33 +08:00
|
|
|
TemplateSpecializationKind VarDecl::getTemplateSpecializationKind() const {
|
2010-02-01 06:27:38 +08:00
|
|
|
if (MemberSpecializationInfo *MSI = getMemberSpecializationInfo())
|
2009-10-08 15:24:58 +08:00
|
|
|
return MSI->getTemplateSpecializationKind();
|
|
|
|
|
|
|
|
return TSK_Undeclared;
|
|
|
|
}
|
|
|
|
|
2009-10-15 05:29:40 +08:00
|
|
|
MemberSpecializationInfo *VarDecl::getMemberSpecializationInfo() const {
|
2009-10-13 04:18:28 +08:00
|
|
|
return getASTContext().getInstantiatedFromStaticDataMember(this);
|
|
|
|
}
|
|
|
|
|
2009-10-16 01:21:20 +08:00
|
|
|
void VarDecl::setTemplateSpecializationKind(TemplateSpecializationKind TSK,
|
|
|
|
SourceLocation PointOfInstantiation) {
|
2009-10-13 04:18:28 +08:00
|
|
|
MemberSpecializationInfo *MSI = getMemberSpecializationInfo();
|
2009-10-08 15:24:58 +08:00
|
|
|
assert(MSI && "Not an instantiated static data member?");
|
|
|
|
MSI->setTemplateSpecializationKind(TSK);
|
2009-10-16 01:21:20 +08:00
|
|
|
if (TSK != TSK_ExplicitSpecialization &&
|
|
|
|
PointOfInstantiation.isValid() &&
|
|
|
|
MSI->getPointOfInstantiation().isInvalid())
|
|
|
|
MSI->setPointOfInstantiation(PointOfInstantiation);
|
2009-07-25 04:34:43 +08:00
|
|
|
}
|
|
|
|
|
2010-01-27 06:01:41 +08:00
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
// ParmVarDecl Implementation
|
|
|
|
//===----------------------------------------------------------------------===//
|
2009-03-11 07:43:53 +08:00
|
|
|
|
2010-01-27 06:01:41 +08:00
|
|
|
ParmVarDecl *ParmVarDecl::Create(ASTContext &C, DeclContext *DC,
|
|
|
|
SourceLocation L, IdentifierInfo *Id,
|
|
|
|
QualType T, TypeSourceInfo *TInfo,
|
|
|
|
StorageClass S, Expr *DefArg) {
|
|
|
|
return new (C) ParmVarDecl(ParmVar, DC, L, Id, T, TInfo, S, DefArg);
|
2009-03-11 07:43:53 +08:00
|
|
|
}
|
|
|
|
|
2010-01-27 06:01:41 +08:00
|
|
|
Expr *ParmVarDecl::getDefaultArg() {
|
|
|
|
assert(!hasUnparsedDefaultArg() && "Default argument is not yet parsed!");
|
|
|
|
assert(!hasUninstantiatedDefaultArg() &&
|
|
|
|
"Default argument is not yet instantiated!");
|
|
|
|
|
|
|
|
Expr *Arg = getInit();
|
|
|
|
if (CXXExprWithTemporaries *E = dyn_cast_or_null<CXXExprWithTemporaries>(Arg))
|
|
|
|
return E->getSubExpr();
|
|
|
|
|
|
|
|
return Arg;
|
|
|
|
}
|
|
|
|
|
|
|
|
unsigned ParmVarDecl::getNumDefaultArgTemporaries() const {
|
|
|
|
if (const CXXExprWithTemporaries *E =
|
|
|
|
dyn_cast<CXXExprWithTemporaries>(getInit()))
|
|
|
|
return E->getNumTemporaries();
|
2009-03-11 07:43:53 +08:00
|
|
|
|
2009-07-14 11:20:21 +08:00
|
|
|
return 0;
|
2009-03-11 07:43:53 +08:00
|
|
|
}
|
|
|
|
|
2010-01-27 06:01:41 +08:00
|
|
|
CXXTemporary *ParmVarDecl::getDefaultArgTemporary(unsigned i) {
|
|
|
|
assert(getNumDefaultArgTemporaries() &&
|
|
|
|
"Default arguments does not have any temporaries!");
|
|
|
|
|
|
|
|
CXXExprWithTemporaries *E = cast<CXXExprWithTemporaries>(getInit());
|
|
|
|
return E->getTemporary(i);
|
|
|
|
}
|
|
|
|
|
|
|
|
SourceRange ParmVarDecl::getDefaultArgRange() const {
|
|
|
|
if (const Expr *E = getInit())
|
|
|
|
return E->getSourceRange();
|
|
|
|
|
|
|
|
if (hasUninstantiatedDefaultArg())
|
|
|
|
return getUninstantiatedDefaultArg()->getSourceRange();
|
|
|
|
|
|
|
|
return SourceRange();
|
2009-07-06 06:21:56 +08:00
|
|
|
}
|
|
|
|
|
2008-11-10 07:41:00 +08:00
|
|
|
//===----------------------------------------------------------------------===//
|
2008-03-31 08:36:02 +08:00
|
|
|
// FunctionDecl Implementation
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
|
2008-05-20 08:43:19 +08:00
|
|
|
void FunctionDecl::Destroy(ASTContext& C) {
|
2009-04-18 08:07:54 +08:00
|
|
|
if (Body && Body.isOffset())
|
|
|
|
Body.get(C.getExternalSource())->Destroy(C);
|
2008-05-20 11:56:00 +08:00
|
|
|
|
|
|
|
for (param_iterator I=param_begin(), E=param_end(); I!=E; ++I)
|
|
|
|
(*I)->Destroy(C);
|
2009-01-19 03:57:27 +08:00
|
|
|
|
2009-10-08 07:56:10 +08:00
|
|
|
FunctionTemplateSpecializationInfo *FTSInfo
|
|
|
|
= TemplateOrSpecialization.dyn_cast<FunctionTemplateSpecializationInfo*>();
|
|
|
|
if (FTSInfo)
|
|
|
|
C.Deallocate(FTSInfo);
|
|
|
|
|
|
|
|
MemberSpecializationInfo *MSInfo
|
|
|
|
= TemplateOrSpecialization.dyn_cast<MemberSpecializationInfo*>();
|
|
|
|
if (MSInfo)
|
|
|
|
C.Deallocate(MSInfo);
|
|
|
|
|
2009-01-28 05:25:57 +08:00
|
|
|
C.Deallocate(ParamInfo);
|
2009-01-19 03:57:27 +08:00
|
|
|
|
2010-03-15 18:12:16 +08:00
|
|
|
DeclaratorDecl::Destroy(C);
|
2008-05-20 08:43:19 +08:00
|
|
|
}
|
|
|
|
|
2009-09-11 14:45:03 +08:00
|
|
|
void FunctionDecl::getNameForDiagnostic(std::string &S,
|
|
|
|
const PrintingPolicy &Policy,
|
|
|
|
bool Qualified) const {
|
|
|
|
NamedDecl::getNameForDiagnostic(S, Policy, Qualified);
|
|
|
|
const TemplateArgumentList *TemplateArgs = getTemplateSpecializationArgs();
|
|
|
|
if (TemplateArgs)
|
|
|
|
S += TemplateSpecializationType::PrintTemplateArgumentList(
|
|
|
|
TemplateArgs->getFlatArgumentList(),
|
|
|
|
TemplateArgs->flat_size(),
|
|
|
|
Policy);
|
|
|
|
|
|
|
|
}
|
2008-05-20 08:43:19 +08:00
|
|
|
|
2009-06-30 10:35:26 +08:00
|
|
|
Stmt *FunctionDecl::getBody(const FunctionDecl *&Definition) const {
|
2009-07-14 11:20:21 +08:00
|
|
|
for (redecl_iterator I = redecls_begin(), E = redecls_end(); I != E; ++I) {
|
|
|
|
if (I->Body) {
|
|
|
|
Definition = *I;
|
|
|
|
return I->Body.get(getASTContext().getExternalSource());
|
2008-04-21 10:02:58 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
2007-01-21 15:42:07 +08:00
|
|
|
}
|
|
|
|
|
2009-06-20 16:09:14 +08:00
|
|
|
void FunctionDecl::setBody(Stmt *B) {
|
|
|
|
Body = B;
|
2009-06-23 01:13:31 +08:00
|
|
|
if (B)
|
2009-06-20 16:09:14 +08:00
|
|
|
EndRangeLoc = B->getLocEnd();
|
|
|
|
}
|
|
|
|
|
2009-09-12 08:17:51 +08:00
|
|
|
bool FunctionDecl::isMain() const {
|
|
|
|
ASTContext &Context = getASTContext();
|
2009-08-15 10:09:25 +08:00
|
|
|
return !Context.getLangOptions().Freestanding &&
|
|
|
|
getDeclContext()->getLookupContext()->isTranslationUnit() &&
|
2009-02-24 09:23:02 +08:00
|
|
|
getIdentifier() && getIdentifier()->isStr("main");
|
|
|
|
}
|
|
|
|
|
2009-09-12 08:17:51 +08:00
|
|
|
bool FunctionDecl::isExternC() const {
|
|
|
|
ASTContext &Context = getASTContext();
|
2009-03-02 08:19:53 +08:00
|
|
|
// In C, any non-static, non-overloadable function has external
|
|
|
|
// linkage.
|
|
|
|
if (!Context.getLangOptions().CPlusPlus)
|
2009-06-30 10:34:44 +08:00
|
|
|
return getStorageClass() != Static && !getAttr<OverloadableAttr>();
|
2009-03-02 08:19:53 +08:00
|
|
|
|
2009-09-09 23:08:12 +08:00
|
|
|
for (const DeclContext *DC = getDeclContext(); !DC->isTranslationUnit();
|
2009-03-02 08:19:53 +08:00
|
|
|
DC = DC->getParent()) {
|
|
|
|
if (const LinkageSpecDecl *Linkage = dyn_cast<LinkageSpecDecl>(DC)) {
|
|
|
|
if (Linkage->getLanguage() == LinkageSpecDecl::lang_c)
|
2009-09-09 23:08:12 +08:00
|
|
|
return getStorageClass() != Static &&
|
2009-06-30 10:34:44 +08:00
|
|
|
!getAttr<OverloadableAttr>();
|
2009-03-02 08:19:53 +08:00
|
|
|
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2009-04-01 00:35:03 +08:00
|
|
|
bool FunctionDecl::isGlobal() const {
|
|
|
|
if (const CXXMethodDecl *Method = dyn_cast<CXXMethodDecl>(this))
|
|
|
|
return Method->isStatic();
|
|
|
|
|
|
|
|
if (getStorageClass() == Static)
|
|
|
|
return false;
|
|
|
|
|
2009-09-09 23:08:12 +08:00
|
|
|
for (const DeclContext *DC = getDeclContext();
|
2009-04-01 00:35:03 +08:00
|
|
|
DC->isNamespace();
|
|
|
|
DC = DC->getParent()) {
|
|
|
|
if (const NamespaceDecl *Namespace = cast<NamespaceDecl>(DC)) {
|
|
|
|
if (!Namespace->getDeclName())
|
|
|
|
return false;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2010-01-27 06:01:41 +08:00
|
|
|
void
|
|
|
|
FunctionDecl::setPreviousDeclaration(FunctionDecl *PrevDecl) {
|
|
|
|
redeclarable_base::setPreviousDeclaration(PrevDecl);
|
|
|
|
|
|
|
|
if (FunctionTemplateDecl *FunTmpl = getDescribedFunctionTemplate()) {
|
|
|
|
FunctionTemplateDecl *PrevFunTmpl
|
|
|
|
= PrevDecl? PrevDecl->getDescribedFunctionTemplate() : 0;
|
|
|
|
assert((!PrevDecl || PrevFunTmpl) && "Function/function template mismatch");
|
|
|
|
FunTmpl->setPreviousDeclaration(PrevFunTmpl);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
const FunctionDecl *FunctionDecl::getCanonicalDecl() const {
|
|
|
|
return getFirstDeclaration();
|
|
|
|
}
|
|
|
|
|
|
|
|
FunctionDecl *FunctionDecl::getCanonicalDecl() {
|
|
|
|
return getFirstDeclaration();
|
|
|
|
}
|
|
|
|
|
Implicitly declare certain C library functions (malloc, strcpy, memmove,
etc.) when we perform name lookup on them. This ensures that we
produce the correct signature for these functions, which has two
practical impacts:
1) When we're supporting the "implicit function declaration" feature
of C99, these functions will be implicitly declared with the right
signature rather than as a function returning "int" with no
prototype. See PR3541 for the reason why this is important (hint:
GCC always predeclares these functions).
2) If users attempt to redeclare one of these library functions with
an incompatible signature, we produce a hard error.
This patch does a little bit of work to give reasonable error
messages. For example, when we hit case #1 we complain that we're
implicitly declaring this function with a specific signature, and then
we give a note that asks the user to include the appropriate header
(e.g., "please include <stdlib.h> or explicitly declare 'malloc'"). In
case #2, we show the type of the implicit builtin that was incorrectly
declared, so the user can see the problem. We could do better here:
for example, when displaying this latter error message we say
something like:
'strcpy' was implicitly declared here with type 'char *(char *, char
const *)'
but we should really print out a fake code line showing the
declaration, like this:
'strcpy' was implicitly declared here as:
char *strcpy(char *, char const *)
This would also be good for printing built-in candidates with C++
operator overloading.
The set of C library functions supported by this patch includes all
functions from the C99 specification's <stdlib.h> and <string.h> that
(a) are predefined by GCC and (b) have signatures that could cause
codegen issues if they are treated as functions with no prototype
returning and int. Future work could extend this set of functions to
other C library functions that we know about.
llvm-svn: 64504
2009-02-14 07:20:09 +08:00
|
|
|
/// \brief Returns a value indicating whether this function
|
|
|
|
/// corresponds to a builtin function.
|
|
|
|
///
|
|
|
|
/// The function corresponds to a built-in function if it is
|
|
|
|
/// declared at translation scope or within an extern "C" block and
|
|
|
|
/// its name matches with the name of a builtin. The returned value
|
|
|
|
/// will be 0 for functions that do not correspond to a builtin, a
|
2009-09-09 23:08:12 +08:00
|
|
|
/// value of type \c Builtin::ID if in the target-independent range
|
Implicitly declare certain C library functions (malloc, strcpy, memmove,
etc.) when we perform name lookup on them. This ensures that we
produce the correct signature for these functions, which has two
practical impacts:
1) When we're supporting the "implicit function declaration" feature
of C99, these functions will be implicitly declared with the right
signature rather than as a function returning "int" with no
prototype. See PR3541 for the reason why this is important (hint:
GCC always predeclares these functions).
2) If users attempt to redeclare one of these library functions with
an incompatible signature, we produce a hard error.
This patch does a little bit of work to give reasonable error
messages. For example, when we hit case #1 we complain that we're
implicitly declaring this function with a specific signature, and then
we give a note that asks the user to include the appropriate header
(e.g., "please include <stdlib.h> or explicitly declare 'malloc'"). In
case #2, we show the type of the implicit builtin that was incorrectly
declared, so the user can see the problem. We could do better here:
for example, when displaying this latter error message we say
something like:
'strcpy' was implicitly declared here with type 'char *(char *, char
const *)'
but we should really print out a fake code line showing the
declaration, like this:
'strcpy' was implicitly declared here as:
char *strcpy(char *, char const *)
This would also be good for printing built-in candidates with C++
operator overloading.
The set of C library functions supported by this patch includes all
functions from the C99 specification's <stdlib.h> and <string.h> that
(a) are predefined by GCC and (b) have signatures that could cause
codegen issues if they are treated as functions with no prototype
returning and int. Future work could extend this set of functions to
other C library functions that we know about.
llvm-svn: 64504
2009-02-14 07:20:09 +08:00
|
|
|
/// \c [1,Builtin::First), or a target-specific builtin value.
|
2009-09-12 08:22:50 +08:00
|
|
|
unsigned FunctionDecl::getBuiltinID() const {
|
|
|
|
ASTContext &Context = getASTContext();
|
2009-02-15 02:57:46 +08:00
|
|
|
if (!getIdentifier() || !getIdentifier()->getBuiltinID())
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
unsigned BuiltinID = getIdentifier()->getBuiltinID();
|
|
|
|
if (!Context.BuiltinInfo.isPredefinedLibFunction(BuiltinID))
|
|
|
|
return BuiltinID;
|
|
|
|
|
|
|
|
// This function has the name of a known C library
|
|
|
|
// function. Determine whether it actually refers to the C library
|
|
|
|
// function or whether it just has the same name.
|
|
|
|
|
2009-02-17 11:23:10 +08:00
|
|
|
// If this is a static function, it's not a builtin.
|
|
|
|
if (getStorageClass() == Static)
|
|
|
|
return 0;
|
|
|
|
|
2009-02-15 02:57:46 +08:00
|
|
|
// If this function is at translation-unit scope and we're not in
|
|
|
|
// C++, it refers to the C library function.
|
|
|
|
if (!Context.getLangOptions().CPlusPlus &&
|
|
|
|
getDeclContext()->isTranslationUnit())
|
|
|
|
return BuiltinID;
|
|
|
|
|
|
|
|
// If the function is in an extern "C" linkage specification and is
|
|
|
|
// not marked "overloadable", it's the real function.
|
|
|
|
if (isa<LinkageSpecDecl>(getDeclContext()) &&
|
2009-09-09 23:08:12 +08:00
|
|
|
cast<LinkageSpecDecl>(getDeclContext())->getLanguage()
|
2009-02-15 02:57:46 +08:00
|
|
|
== LinkageSpecDecl::lang_c &&
|
2009-06-30 10:34:44 +08:00
|
|
|
!getAttr<OverloadableAttr>())
|
2009-02-15 02:57:46 +08:00
|
|
|
return BuiltinID;
|
|
|
|
|
|
|
|
// Not a builtin
|
Implicitly declare certain C library functions (malloc, strcpy, memmove,
etc.) when we perform name lookup on them. This ensures that we
produce the correct signature for these functions, which has two
practical impacts:
1) When we're supporting the "implicit function declaration" feature
of C99, these functions will be implicitly declared with the right
signature rather than as a function returning "int" with no
prototype. See PR3541 for the reason why this is important (hint:
GCC always predeclares these functions).
2) If users attempt to redeclare one of these library functions with
an incompatible signature, we produce a hard error.
This patch does a little bit of work to give reasonable error
messages. For example, when we hit case #1 we complain that we're
implicitly declaring this function with a specific signature, and then
we give a note that asks the user to include the appropriate header
(e.g., "please include <stdlib.h> or explicitly declare 'malloc'"). In
case #2, we show the type of the implicit builtin that was incorrectly
declared, so the user can see the problem. We could do better here:
for example, when displaying this latter error message we say
something like:
'strcpy' was implicitly declared here with type 'char *(char *, char
const *)'
but we should really print out a fake code line showing the
declaration, like this:
'strcpy' was implicitly declared here as:
char *strcpy(char *, char const *)
This would also be good for printing built-in candidates with C++
operator overloading.
The set of C library functions supported by this patch includes all
functions from the C99 specification's <stdlib.h> and <string.h> that
(a) are predefined by GCC and (b) have signatures that could cause
codegen issues if they are treated as functions with no prototype
returning and int. Future work could extend this set of functions to
other C library functions that we know about.
llvm-svn: 64504
2009-02-14 07:20:09 +08:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2009-04-25 14:03:53 +08:00
|
|
|
/// getNumParams - Return the number of parameters this function must have
|
2009-04-25 14:12:16 +08:00
|
|
|
/// based on its FunctionType. This is the length of the PararmInfo array
|
2009-04-25 14:03:53 +08:00
|
|
|
/// after it has been created.
|
|
|
|
unsigned FunctionDecl::getNumParams() const {
|
2009-09-22 07:43:11 +08:00
|
|
|
const FunctionType *FT = getType()->getAs<FunctionType>();
|
2009-02-27 07:50:07 +08:00
|
|
|
if (isa<FunctionNoProtoType>(FT))
|
2008-03-15 13:43:15 +08:00
|
|
|
return 0;
|
2009-02-27 07:50:07 +08:00
|
|
|
return cast<FunctionProtoType>(FT)->getNumArgs();
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2007-01-21 15:42:07 +08:00
|
|
|
}
|
|
|
|
|
2010-02-11 09:19:42 +08:00
|
|
|
void FunctionDecl::setParams(ParmVarDecl **NewParamInfo, unsigned NumParams) {
|
2007-01-21 15:42:07 +08:00
|
|
|
assert(ParamInfo == 0 && "Already has param info!");
|
2009-04-25 14:12:16 +08:00
|
|
|
assert(NumParams == getNumParams() && "Parameter count mismatch!");
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2007-01-22 03:04:10 +08:00
|
|
|
// Zero params -> null pointer.
|
|
|
|
if (NumParams) {
|
2010-02-11 09:19:42 +08:00
|
|
|
void *Mem = getASTContext().Allocate(sizeof(ParmVarDecl*)*NumParams);
|
2009-01-14 08:42:25 +08:00
|
|
|
ParamInfo = new (Mem) ParmVarDecl*[NumParams];
|
2007-06-14 04:44:40 +08:00
|
|
|
memcpy(ParamInfo, NewParamInfo, sizeof(ParmVarDecl*)*NumParams);
|
2009-06-20 16:09:14 +08:00
|
|
|
|
2009-06-23 08:42:00 +08:00
|
|
|
// Update source range. The check below allows us to set EndRangeLoc before
|
|
|
|
// setting the parameters.
|
2009-06-23 08:42:15 +08:00
|
|
|
if (EndRangeLoc.isInvalid() || EndRangeLoc == getLocation())
|
2009-06-20 16:09:14 +08:00
|
|
|
EndRangeLoc = NewParamInfo[NumParams-1]->getLocEnd();
|
2007-01-22 03:04:10 +08:00
|
|
|
}
|
2007-01-21 15:42:07 +08:00
|
|
|
}
|
2007-01-25 12:52:46 +08:00
|
|
|
|
2008-04-10 10:22:51 +08:00
|
|
|
/// getMinRequiredArguments - Returns the minimum number of arguments
|
|
|
|
/// needed to call this function. This may be fewer than the number of
|
|
|
|
/// function parameters, if some of the parameters have default
|
2008-04-13 07:52:44 +08:00
|
|
|
/// arguments (in C++).
|
2008-04-10 10:22:51 +08:00
|
|
|
unsigned FunctionDecl::getMinRequiredArguments() const {
|
|
|
|
unsigned NumRequiredArgs = getNumParams();
|
|
|
|
while (NumRequiredArgs > 0
|
2009-06-06 12:14:07 +08:00
|
|
|
&& getParamDecl(NumRequiredArgs-1)->hasDefaultArg())
|
2008-04-10 10:22:51 +08:00
|
|
|
--NumRequiredArgs;
|
|
|
|
|
|
|
|
return NumRequiredArgs;
|
|
|
|
}
|
|
|
|
|
2009-10-28 05:11:48 +08:00
|
|
|
bool FunctionDecl::isInlined() const {
|
2009-12-05 06:35:50 +08:00
|
|
|
// FIXME: This is not enough. Consider:
|
|
|
|
//
|
|
|
|
// inline void f();
|
|
|
|
// void f() { }
|
|
|
|
//
|
|
|
|
// f is inlined, but does not have inline specified.
|
|
|
|
// To fix this we should add an 'inline' flag to FunctionDecl.
|
|
|
|
if (isInlineSpecified())
|
2009-10-28 07:26:40 +08:00
|
|
|
return true;
|
2009-12-05 06:35:50 +08:00
|
|
|
|
|
|
|
if (isa<CXXMethodDecl>(this)) {
|
|
|
|
if (!isOutOfLine() || getCanonicalDecl()->isInlineSpecified())
|
|
|
|
return true;
|
|
|
|
}
|
2009-10-28 07:26:40 +08:00
|
|
|
|
|
|
|
switch (getTemplateSpecializationKind()) {
|
|
|
|
case TSK_Undeclared:
|
|
|
|
case TSK_ExplicitSpecialization:
|
|
|
|
return false;
|
|
|
|
|
|
|
|
case TSK_ImplicitInstantiation:
|
|
|
|
case TSK_ExplicitInstantiationDeclaration:
|
|
|
|
case TSK_ExplicitInstantiationDefinition:
|
|
|
|
// Handle below.
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
const FunctionDecl *PatternDecl = getTemplateInstantiationPattern();
|
|
|
|
Stmt *Pattern = 0;
|
|
|
|
if (PatternDecl)
|
|
|
|
Pattern = PatternDecl->getBody(PatternDecl);
|
|
|
|
|
|
|
|
if (Pattern && PatternDecl)
|
|
|
|
return PatternDecl->isInlined();
|
|
|
|
|
|
|
|
return false;
|
2009-10-28 05:11:48 +08:00
|
|
|
}
|
|
|
|
|
2009-10-28 07:26:40 +08:00
|
|
|
/// \brief For an inline function definition in C or C++, determine whether the
|
2009-09-13 15:46:26 +08:00
|
|
|
/// definition will be externally visible.
|
|
|
|
///
|
|
|
|
/// Inline function definitions are always available for inlining optimizations.
|
|
|
|
/// However, depending on the language dialect, declaration specifiers, and
|
|
|
|
/// attributes, the definition of an inline function may or may not be
|
|
|
|
/// "externally" visible to other translation units in the program.
|
|
|
|
///
|
|
|
|
/// In C99, inline definitions are not externally visible by default. However,
|
2010-01-06 10:05:39 +08:00
|
|
|
/// if even one of the global-scope declarations is marked "extern inline", the
|
2009-09-13 15:46:26 +08:00
|
|
|
/// inline definition becomes externally visible (C99 6.7.4p6).
|
|
|
|
///
|
|
|
|
/// In GNU89 mode, or if the gnu_inline attribute is attached to the function
|
|
|
|
/// definition, we use the GNU semantics for inline, which are nearly the
|
|
|
|
/// opposite of C99 semantics. In particular, "inline" by itself will create
|
|
|
|
/// an externally visible symbol, but "extern inline" will not create an
|
|
|
|
/// externally visible symbol.
|
|
|
|
bool FunctionDecl::isInlineDefinitionExternallyVisible() const {
|
|
|
|
assert(isThisDeclarationADefinition() && "Must have the function definition");
|
2009-10-28 05:11:48 +08:00
|
|
|
assert(isInlined() && "Function must be inline");
|
2009-10-28 07:26:40 +08:00
|
|
|
ASTContext &Context = getASTContext();
|
2009-09-13 15:46:26 +08:00
|
|
|
|
2009-10-28 07:26:40 +08:00
|
|
|
if (!Context.getLangOptions().C99 || hasAttr<GNUInlineAttr>()) {
|
2009-09-13 15:46:26 +08:00
|
|
|
// GNU inline semantics. Based on a number of examples, we came up with the
|
|
|
|
// following heuristic: if the "inline" keyword is present on a
|
|
|
|
// declaration of the function but "extern" is not present on that
|
|
|
|
// declaration, then the symbol is externally visible. Otherwise, the GNU
|
|
|
|
// "extern inline" semantics applies and the symbol is not externally
|
|
|
|
// visible.
|
|
|
|
for (redecl_iterator Redecl = redecls_begin(), RedeclEnd = redecls_end();
|
|
|
|
Redecl != RedeclEnd;
|
|
|
|
++Redecl) {
|
2009-10-28 05:01:01 +08:00
|
|
|
if (Redecl->isInlineSpecified() && Redecl->getStorageClass() != Extern)
|
2009-09-13 15:46:26 +08:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
// GNU "extern inline" semantics; no externally visible symbol.
|
2009-04-28 14:37:30 +08:00
|
|
|
return false;
|
2009-09-13 15:46:26 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
// C99 6.7.4p6:
|
|
|
|
// [...] If all of the file scope declarations for a function in a
|
|
|
|
// translation unit include the inline function specifier without extern,
|
|
|
|
// then the definition in that translation unit is an inline definition.
|
|
|
|
for (redecl_iterator Redecl = redecls_begin(), RedeclEnd = redecls_end();
|
|
|
|
Redecl != RedeclEnd;
|
|
|
|
++Redecl) {
|
|
|
|
// Only consider file-scope declarations in this test.
|
|
|
|
if (!Redecl->getLexicalDeclContext()->isTranslationUnit())
|
|
|
|
continue;
|
|
|
|
|
2009-10-28 05:01:01 +08:00
|
|
|
if (!Redecl->isInlineSpecified() || Redecl->getStorageClass() == Extern)
|
2009-09-13 15:46:26 +08:00
|
|
|
return true; // Not an inline definition
|
|
|
|
}
|
|
|
|
|
|
|
|
// C99 6.7.4p6:
|
|
|
|
// An inline definition does not provide an external definition for the
|
|
|
|
// function, and does not forbid an external definition in another
|
|
|
|
// translation unit.
|
2009-04-28 14:37:30 +08:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2008-11-07 06:13:31 +08:00
|
|
|
/// getOverloadedOperator - Which C++ overloaded operator this
|
|
|
|
/// function represents, if any.
|
|
|
|
OverloadedOperatorKind FunctionDecl::getOverloadedOperator() const {
|
Extend DeclarationName to support C++ overloaded operators, e.g.,
operator+, directly, using the same mechanism as all other special
names.
Removed the "special" identifiers for the overloaded operators from
the identifier table and IdentifierInfo data structure. IdentifierInfo
is back to representing only real identifiers.
Added a new Action, ActOnOperatorFunctionIdExpr, that builds an
expression from an parsed operator-function-id (e.g., "operator
+"). ActOnIdentifierExpr used to do this job, but
operator-function-ids are no longer represented by IdentifierInfo's.
Extended Declarator to store overloaded operator names.
Sema::GetNameForDeclarator now knows how to turn the operator
name into a DeclarationName for the overloaded operator.
Except for (perhaps) consolidating the functionality of
ActOnIdentifier, ActOnOperatorFunctionIdExpr, and
ActOnConversionFunctionExpr into a common routine that builds an
appropriate DeclRefExpr by looking up a DeclarationName, all of the
work on normalizing declaration names should be complete with this
commit.
llvm-svn: 59526
2008-11-18 22:39:36 +08:00
|
|
|
if (getDeclName().getNameKind() == DeclarationName::CXXOperatorName)
|
|
|
|
return getDeclName().getCXXOverloadedOperator();
|
2008-11-07 06:13:31 +08:00
|
|
|
else
|
|
|
|
return OO_None;
|
|
|
|
}
|
|
|
|
|
2010-01-13 17:01:02 +08:00
|
|
|
/// getLiteralIdentifier - The literal suffix identifier this function
|
|
|
|
/// represents, if any.
|
|
|
|
const IdentifierInfo *FunctionDecl::getLiteralIdentifier() const {
|
|
|
|
if (getDeclName().getNameKind() == DeclarationName::CXXLiteralOperatorName)
|
|
|
|
return getDeclName().getCXXLiteralIdentifier();
|
|
|
|
else
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2009-10-08 07:56:10 +08:00
|
|
|
FunctionDecl *FunctionDecl::getInstantiatedFromMemberFunction() const {
|
2009-10-13 04:18:28 +08:00
|
|
|
if (MemberSpecializationInfo *Info = getMemberSpecializationInfo())
|
2009-10-08 07:56:10 +08:00
|
|
|
return cast<FunctionDecl>(Info->getInstantiatedFrom());
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2009-10-13 04:18:28 +08:00
|
|
|
MemberSpecializationInfo *FunctionDecl::getMemberSpecializationInfo() const {
|
|
|
|
return TemplateOrSpecialization.dyn_cast<MemberSpecializationInfo*>();
|
|
|
|
}
|
|
|
|
|
2009-10-08 07:56:10 +08:00
|
|
|
void
|
|
|
|
FunctionDecl::setInstantiationOfMemberFunction(FunctionDecl *FD,
|
|
|
|
TemplateSpecializationKind TSK) {
|
|
|
|
assert(TemplateOrSpecialization.isNull() &&
|
|
|
|
"Member function is already a specialization");
|
|
|
|
MemberSpecializationInfo *Info
|
|
|
|
= new (getASTContext()) MemberSpecializationInfo(FD, TSK);
|
|
|
|
TemplateOrSpecialization = Info;
|
|
|
|
}
|
|
|
|
|
2009-10-28 04:53:28 +08:00
|
|
|
bool FunctionDecl::isImplicitlyInstantiable() const {
|
|
|
|
// If this function already has a definition or is invalid, it can't be
|
|
|
|
// implicitly instantiated.
|
|
|
|
if (isInvalidDecl() || getBody())
|
|
|
|
return false;
|
|
|
|
|
|
|
|
switch (getTemplateSpecializationKind()) {
|
|
|
|
case TSK_Undeclared:
|
|
|
|
case TSK_ExplicitSpecialization:
|
|
|
|
case TSK_ExplicitInstantiationDefinition:
|
|
|
|
return false;
|
|
|
|
|
|
|
|
case TSK_ImplicitInstantiation:
|
|
|
|
return true;
|
|
|
|
|
|
|
|
case TSK_ExplicitInstantiationDeclaration:
|
|
|
|
// Handled below.
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Find the actual template from which we will instantiate.
|
|
|
|
const FunctionDecl *PatternDecl = getTemplateInstantiationPattern();
|
|
|
|
Stmt *Pattern = 0;
|
|
|
|
if (PatternDecl)
|
|
|
|
Pattern = PatternDecl->getBody(PatternDecl);
|
|
|
|
|
|
|
|
// C++0x [temp.explicit]p9:
|
|
|
|
// Except for inline functions, other explicit instantiation declarations
|
|
|
|
// have the effect of suppressing the implicit instantiation of the entity
|
|
|
|
// to which they refer.
|
|
|
|
if (!Pattern || !PatternDecl)
|
|
|
|
return true;
|
|
|
|
|
2009-10-28 05:11:48 +08:00
|
|
|
return PatternDecl->isInlined();
|
2009-10-28 04:53:28 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
FunctionDecl *FunctionDecl::getTemplateInstantiationPattern() const {
|
|
|
|
if (FunctionTemplateDecl *Primary = getPrimaryTemplate()) {
|
|
|
|
while (Primary->getInstantiatedFromMemberTemplate()) {
|
|
|
|
// If we have hit a point where the user provided a specialization of
|
|
|
|
// this template, we're done looking.
|
|
|
|
if (Primary->isMemberSpecialization())
|
|
|
|
break;
|
|
|
|
|
|
|
|
Primary = Primary->getInstantiatedFromMemberTemplate();
|
|
|
|
}
|
|
|
|
|
|
|
|
return Primary->getTemplatedDecl();
|
|
|
|
}
|
|
|
|
|
|
|
|
return getInstantiatedFromMemberFunction();
|
|
|
|
}
|
|
|
|
|
2009-06-30 01:30:29 +08:00
|
|
|
FunctionTemplateDecl *FunctionDecl::getPrimaryTemplate() const {
|
2009-09-09 23:08:12 +08:00
|
|
|
if (FunctionTemplateSpecializationInfo *Info
|
2009-06-30 01:30:29 +08:00
|
|
|
= TemplateOrSpecialization
|
|
|
|
.dyn_cast<FunctionTemplateSpecializationInfo*>()) {
|
2009-06-30 06:39:32 +08:00
|
|
|
return Info->Template.getPointer();
|
2009-06-30 01:30:29 +08:00
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
const TemplateArgumentList *
|
|
|
|
FunctionDecl::getTemplateSpecializationArgs() const {
|
2009-09-09 23:08:12 +08:00
|
|
|
if (FunctionTemplateSpecializationInfo *Info
|
2009-10-14 00:30:37 +08:00
|
|
|
= TemplateOrSpecialization
|
|
|
|
.dyn_cast<FunctionTemplateSpecializationInfo*>()) {
|
2009-06-30 01:30:29 +08:00
|
|
|
return Info->TemplateArguments;
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2009-09-09 23:08:12 +08:00
|
|
|
void
|
2010-02-11 09:19:42 +08:00
|
|
|
FunctionDecl::setFunctionTemplateSpecialization(FunctionTemplateDecl *Template,
|
2009-06-30 04:59:39 +08:00
|
|
|
const TemplateArgumentList *TemplateArgs,
|
2009-09-25 07:14:47 +08:00
|
|
|
void *InsertPos,
|
|
|
|
TemplateSpecializationKind TSK) {
|
|
|
|
assert(TSK != TSK_Undeclared &&
|
|
|
|
"Must specify the type of function template specialization");
|
2009-09-09 23:08:12 +08:00
|
|
|
FunctionTemplateSpecializationInfo *Info
|
2009-06-30 01:30:29 +08:00
|
|
|
= TemplateOrSpecialization.dyn_cast<FunctionTemplateSpecializationInfo*>();
|
2009-06-26 08:10:03 +08:00
|
|
|
if (!Info)
|
2010-02-11 09:19:42 +08:00
|
|
|
Info = new (getASTContext()) FunctionTemplateSpecializationInfo;
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-06-30 04:59:39 +08:00
|
|
|
Info->Function = this;
|
2009-06-30 06:39:32 +08:00
|
|
|
Info->Template.setPointer(Template);
|
2009-09-25 07:14:47 +08:00
|
|
|
Info->Template.setInt(TSK - 1);
|
2009-06-26 08:10:03 +08:00
|
|
|
Info->TemplateArguments = TemplateArgs;
|
|
|
|
TemplateOrSpecialization = Info;
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-06-30 04:59:39 +08:00
|
|
|
// Insert this function template specialization into the set of known
|
2009-09-25 07:14:47 +08:00
|
|
|
// function template specializations.
|
|
|
|
if (InsertPos)
|
|
|
|
Template->getSpecializations().InsertNode(Info, InsertPos);
|
|
|
|
else {
|
|
|
|
// Try to insert the new node. If there is an existing node, remove it
|
|
|
|
// first.
|
|
|
|
FunctionTemplateSpecializationInfo *Existing
|
|
|
|
= Template->getSpecializations().GetOrInsertNode(Info);
|
|
|
|
if (Existing) {
|
|
|
|
Template->getSpecializations().RemoveNode(Existing);
|
|
|
|
Template->getSpecializations().GetOrInsertNode(Info);
|
|
|
|
}
|
|
|
|
}
|
2009-06-26 08:10:03 +08:00
|
|
|
}
|
|
|
|
|
2009-09-05 06:48:11 +08:00
|
|
|
TemplateSpecializationKind FunctionDecl::getTemplateSpecializationKind() const {
|
2009-09-09 23:08:12 +08:00
|
|
|
// For a function template specialization, query the specialization
|
2009-09-05 06:48:11 +08:00
|
|
|
// information object.
|
2009-10-08 07:56:10 +08:00
|
|
|
FunctionTemplateSpecializationInfo *FTSInfo
|
2009-06-30 06:39:32 +08:00
|
|
|
= TemplateOrSpecialization.dyn_cast<FunctionTemplateSpecializationInfo*>();
|
2009-10-08 07:56:10 +08:00
|
|
|
if (FTSInfo)
|
|
|
|
return FTSInfo->getTemplateSpecializationKind();
|
2009-09-05 06:48:11 +08:00
|
|
|
|
2009-10-08 07:56:10 +08:00
|
|
|
MemberSpecializationInfo *MSInfo
|
|
|
|
= TemplateOrSpecialization.dyn_cast<MemberSpecializationInfo*>();
|
|
|
|
if (MSInfo)
|
|
|
|
return MSInfo->getTemplateSpecializationKind();
|
|
|
|
|
|
|
|
return TSK_Undeclared;
|
2009-06-30 06:39:32 +08:00
|
|
|
}
|
|
|
|
|
2009-09-09 23:08:12 +08:00
|
|
|
void
|
2009-10-16 01:21:20 +08:00
|
|
|
FunctionDecl::setTemplateSpecializationKind(TemplateSpecializationKind TSK,
|
|
|
|
SourceLocation PointOfInstantiation) {
|
2009-10-08 07:56:10 +08:00
|
|
|
if (FunctionTemplateSpecializationInfo *FTSInfo
|
|
|
|
= TemplateOrSpecialization.dyn_cast<
|
2009-10-16 01:21:20 +08:00
|
|
|
FunctionTemplateSpecializationInfo*>()) {
|
2009-10-08 07:56:10 +08:00
|
|
|
FTSInfo->setTemplateSpecializationKind(TSK);
|
2009-10-16 01:21:20 +08:00
|
|
|
if (TSK != TSK_ExplicitSpecialization &&
|
|
|
|
PointOfInstantiation.isValid() &&
|
|
|
|
FTSInfo->getPointOfInstantiation().isInvalid())
|
|
|
|
FTSInfo->setPointOfInstantiation(PointOfInstantiation);
|
|
|
|
} else if (MemberSpecializationInfo *MSInfo
|
|
|
|
= TemplateOrSpecialization.dyn_cast<MemberSpecializationInfo*>()) {
|
2009-10-08 07:56:10 +08:00
|
|
|
MSInfo->setTemplateSpecializationKind(TSK);
|
2009-10-16 01:21:20 +08:00
|
|
|
if (TSK != TSK_ExplicitSpecialization &&
|
|
|
|
PointOfInstantiation.isValid() &&
|
|
|
|
MSInfo->getPointOfInstantiation().isInvalid())
|
|
|
|
MSInfo->setPointOfInstantiation(PointOfInstantiation);
|
|
|
|
} else
|
2009-10-08 07:56:10 +08:00
|
|
|
assert(false && "Function cannot have a template specialization kind");
|
2009-06-30 06:39:32 +08:00
|
|
|
}
|
|
|
|
|
2009-10-16 01:21:20 +08:00
|
|
|
SourceLocation FunctionDecl::getPointOfInstantiation() const {
|
|
|
|
if (FunctionTemplateSpecializationInfo *FTSInfo
|
|
|
|
= TemplateOrSpecialization.dyn_cast<
|
|
|
|
FunctionTemplateSpecializationInfo*>())
|
|
|
|
return FTSInfo->getPointOfInstantiation();
|
|
|
|
else if (MemberSpecializationInfo *MSInfo
|
|
|
|
= TemplateOrSpecialization.dyn_cast<MemberSpecializationInfo*>())
|
|
|
|
return MSInfo->getPointOfInstantiation();
|
|
|
|
|
|
|
|
return SourceLocation();
|
|
|
|
}
|
|
|
|
|
2009-09-12 04:15:17 +08:00
|
|
|
bool FunctionDecl::isOutOfLine() const {
|
|
|
|
if (Decl::isOutOfLine())
|
|
|
|
return true;
|
|
|
|
|
|
|
|
// If this function was instantiated from a member function of a
|
|
|
|
// class template, check whether that member function was defined out-of-line.
|
|
|
|
if (FunctionDecl *FD = getInstantiatedFromMemberFunction()) {
|
|
|
|
const FunctionDecl *Definition;
|
|
|
|
if (FD->getBody(Definition))
|
|
|
|
return Definition->isOutOfLine();
|
|
|
|
}
|
|
|
|
|
|
|
|
// If this function was instantiated from a function template,
|
|
|
|
// check whether that function template was defined out-of-line.
|
|
|
|
if (FunctionTemplateDecl *FunTmpl = getPrimaryTemplate()) {
|
|
|
|
const FunctionDecl *Definition;
|
|
|
|
if (FunTmpl->getTemplatedDecl()->getBody(Definition))
|
|
|
|
return Definition->isOutOfLine();
|
|
|
|
}
|
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2010-01-27 06:01:41 +08:00
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
// FieldDecl Implementation
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
|
|
|
|
FieldDecl *FieldDecl::Create(ASTContext &C, DeclContext *DC, SourceLocation L,
|
|
|
|
IdentifierInfo *Id, QualType T,
|
|
|
|
TypeSourceInfo *TInfo, Expr *BW, bool Mutable) {
|
|
|
|
return new (C) FieldDecl(Decl::Field, DC, L, Id, T, TInfo, BW, Mutable);
|
|
|
|
}
|
|
|
|
|
|
|
|
bool FieldDecl::isAnonymousStructOrUnion() const {
|
|
|
|
if (!isImplicit() || getDeclName())
|
|
|
|
return false;
|
|
|
|
|
|
|
|
if (const RecordType *Record = getType()->getAs<RecordType>())
|
|
|
|
return Record->getDecl()->isAnonymousStructOrUnion();
|
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
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-01-07 08:43:41 +08:00
|
|
|
// TagDecl Implementation
|
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
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
|
2010-03-15 18:12:16 +08:00
|
|
|
void TagDecl::Destroy(ASTContext &C) {
|
|
|
|
if (hasExtInfo())
|
|
|
|
C.Deallocate(getExtInfo());
|
|
|
|
TypeDecl::Destroy(C);
|
|
|
|
}
|
|
|
|
|
2009-07-14 11:17:17 +08:00
|
|
|
SourceRange TagDecl::getSourceRange() const {
|
|
|
|
SourceLocation E = RBraceLoc.isValid() ? RBraceLoc : getLocation();
|
2009-07-21 22:46:17 +08:00
|
|
|
return SourceRange(TagKeywordLoc, E);
|
2009-07-14 11:17:17 +08:00
|
|
|
}
|
|
|
|
|
2009-07-18 08:34:07 +08:00
|
|
|
TagDecl* TagDecl::getCanonicalDecl() {
|
2009-07-30 07:36:44 +08:00
|
|
|
return getFirstDeclaration();
|
2009-07-18 08:34:07 +08:00
|
|
|
}
|
|
|
|
|
2009-01-17 08:42:38 +08:00
|
|
|
void TagDecl::startDefinition() {
|
2009-07-30 07:36:44 +08:00
|
|
|
if (TagType *TagT = const_cast<TagType *>(TypeForDecl->getAs<TagType>())) {
|
|
|
|
TagT->decl.setPointer(this);
|
|
|
|
TagT->decl.setInt(1);
|
|
|
|
}
|
2010-02-05 06:26:26 +08:00
|
|
|
|
|
|
|
if (isa<CXXRecordDecl>(this)) {
|
|
|
|
CXXRecordDecl *D = cast<CXXRecordDecl>(this);
|
|
|
|
struct CXXRecordDecl::DefinitionData *Data =
|
|
|
|
new (getASTContext()) struct CXXRecordDecl::DefinitionData(D);
|
2010-03-27 05:56:38 +08:00
|
|
|
for (redecl_iterator I = redecls_begin(), E = redecls_end(); I != E; ++I)
|
|
|
|
cast<CXXRecordDecl>(*I)->DefinitionData = Data;
|
2010-02-05 06:26:26 +08:00
|
|
|
}
|
2009-01-17 08:42:38 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
void TagDecl::completeDefinition() {
|
2010-02-05 09:33:36 +08:00
|
|
|
assert((!isa<CXXRecordDecl>(this) ||
|
|
|
|
cast<CXXRecordDecl>(this)->hasDefinition()) &&
|
|
|
|
"definition completed but not started");
|
|
|
|
|
2009-01-17 08:42:38 +08:00
|
|
|
IsDefinition = true;
|
2009-07-30 07:36:44 +08:00
|
|
|
if (TagType *TagT = const_cast<TagType *>(TypeForDecl->getAs<TagType>())) {
|
|
|
|
assert(TagT->decl.getPointer() == this &&
|
|
|
|
"Attempt to redefine a tag definition?");
|
|
|
|
TagT->decl.setInt(0);
|
|
|
|
}
|
2009-01-17 08:42:38 +08:00
|
|
|
}
|
|
|
|
|
2010-02-11 09:04:33 +08:00
|
|
|
TagDecl* TagDecl::getDefinition() const {
|
2009-07-30 07:36:44 +08:00
|
|
|
if (isDefinition())
|
|
|
|
return const_cast<TagDecl *>(this);
|
2009-09-09 23:08:12 +08:00
|
|
|
|
|
|
|
for (redecl_iterator R = redecls_begin(), REnd = redecls_end();
|
2009-07-30 07:36:44 +08:00
|
|
|
R != REnd; ++R)
|
|
|
|
if (R->isDefinition())
|
|
|
|
return *R;
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-07-30 07:36:44 +08:00
|
|
|
return 0;
|
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-09-04 09:14:41 +08:00
|
|
|
TagDecl::TagKind TagDecl::getTagKindForTypeSpec(unsigned TypeSpec) {
|
|
|
|
switch (TypeSpec) {
|
2009-12-12 13:05:38 +08:00
|
|
|
default: llvm_unreachable("unexpected type specifier");
|
2009-09-04 09:14:41 +08:00
|
|
|
case DeclSpec::TST_struct: return TK_struct;
|
|
|
|
case DeclSpec::TST_class: return TK_class;
|
|
|
|
case DeclSpec::TST_union: return TK_union;
|
|
|
|
case DeclSpec::TST_enum: return TK_enum;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2010-03-15 18:12:16 +08:00
|
|
|
void TagDecl::setQualifierInfo(NestedNameSpecifier *Qualifier,
|
|
|
|
SourceRange QualifierRange) {
|
|
|
|
if (Qualifier) {
|
|
|
|
// Make sure the extended qualifier info is allocated.
|
|
|
|
if (!hasExtInfo())
|
|
|
|
TypedefDeclOrQualifier = new (getASTContext()) ExtInfo;
|
|
|
|
// Set qualifier info.
|
|
|
|
getExtInfo()->NNS = Qualifier;
|
|
|
|
getExtInfo()->NNSRange = QualifierRange;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
// Here Qualifier == 0, i.e., we are removing the qualifier (if any).
|
|
|
|
assert(QualifierRange.isInvalid());
|
|
|
|
if (hasExtInfo()) {
|
|
|
|
getASTContext().Deallocate(getExtInfo());
|
|
|
|
TypedefDeclOrQualifier = (TypedefDecl*) 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2010-01-27 06:01:41 +08:00
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
// EnumDecl Implementation
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
|
|
|
|
EnumDecl *EnumDecl::Create(ASTContext &C, DeclContext *DC, SourceLocation L,
|
|
|
|
IdentifierInfo *Id, SourceLocation TKL,
|
|
|
|
EnumDecl *PrevDecl) {
|
|
|
|
EnumDecl *Enum = new (C) EnumDecl(DC, L, Id, PrevDecl, TKL);
|
|
|
|
C.getTypeDeclType(Enum, PrevDecl);
|
|
|
|
return Enum;
|
|
|
|
}
|
|
|
|
|
|
|
|
void EnumDecl::Destroy(ASTContext& C) {
|
2010-03-15 18:12:16 +08:00
|
|
|
TagDecl::Destroy(C);
|
2010-01-27 06:01:41 +08:00
|
|
|
}
|
|
|
|
|
2010-02-11 09:19:42 +08:00
|
|
|
void EnumDecl::completeDefinition(QualType NewType,
|
2010-01-27 06:01:41 +08:00
|
|
|
QualType NewPromotionType) {
|
|
|
|
assert(!isDefinition() && "Cannot redefine enums!");
|
|
|
|
IntegerType = NewType;
|
|
|
|
PromotionType = NewPromotionType;
|
|
|
|
TagDecl::completeDefinition();
|
|
|
|
}
|
|
|
|
|
2008-03-31 08:36:02 +08:00
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
// RecordDecl Implementation
|
|
|
|
//===----------------------------------------------------------------------===//
|
2007-01-25 12:52:46 +08:00
|
|
|
|
2008-10-15 08:42:39 +08:00
|
|
|
RecordDecl::RecordDecl(Kind DK, TagKind TK, DeclContext *DC, SourceLocation L,
|
2009-07-30 07:36:44 +08:00
|
|
|
IdentifierInfo *Id, RecordDecl *PrevDecl,
|
|
|
|
SourceLocation TKL)
|
|
|
|
: TagDecl(DK, TK, DC, L, Id, PrevDecl, TKL) {
|
2008-09-03 05:12:32 +08:00
|
|
|
HasFlexibleArrayMember = false;
|
2009-01-07 08:43:41 +08:00
|
|
|
AnonymousStructOrUnion = false;
|
2009-07-08 09:18:33 +08:00
|
|
|
HasObjectMember = false;
|
2008-09-03 05:12:32 +08:00
|
|
|
assert(classof(static_cast<Decl*>(this)) && "Invalid Kind!");
|
|
|
|
}
|
|
|
|
|
|
|
|
RecordDecl *RecordDecl::Create(ASTContext &C, TagKind TK, DeclContext *DC,
|
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
|
|
|
SourceLocation L, IdentifierInfo *Id,
|
2009-07-21 22:46:17 +08:00
|
|
|
SourceLocation TKL, RecordDecl* PrevDecl) {
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-07-30 07:36:44 +08:00
|
|
|
RecordDecl* R = new (C) RecordDecl(Record, TK, DC, L, Id, PrevDecl, TKL);
|
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
|
|
|
C.getTypeDeclType(R, PrevDecl);
|
|
|
|
return R;
|
2008-09-03 05:12:32 +08:00
|
|
|
}
|
|
|
|
|
2008-08-08 22:08:55 +08:00
|
|
|
RecordDecl::~RecordDecl() {
|
|
|
|
}
|
|
|
|
|
|
|
|
void RecordDecl::Destroy(ASTContext& C) {
|
|
|
|
TagDecl::Destroy(C);
|
|
|
|
}
|
|
|
|
|
2009-03-25 23:59:44 +08:00
|
|
|
bool RecordDecl::isInjectedClassName() const {
|
2009-09-09 23:08:12 +08:00
|
|
|
return isImplicit() && getDeclName() && getDeclContext()->isRecord() &&
|
2009-03-25 23:59:44 +08:00
|
|
|
cast<RecordDecl>(getDeclContext())->getDeclName() == getDeclName();
|
|
|
|
}
|
|
|
|
|
2008-12-12 00:49:14 +08:00
|
|
|
/// completeDefinition - Notes that the definition of this type is now
|
|
|
|
/// complete.
|
2010-02-11 09:19:42 +08:00
|
|
|
void RecordDecl::completeDefinition() {
|
2007-01-25 12:52:46 +08:00
|
|
|
assert(!isDefinition() && "Cannot redefine record!");
|
2009-01-17 08:42:38 +08:00
|
|
|
TagDecl::completeDefinition();
|
2007-01-25 12:52:46 +08:00
|
|
|
}
|
2007-03-27 07:09:51 +08:00
|
|
|
|
2008-10-09 01:01:13 +08:00
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
// BlockDecl Implementation
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
|
|
|
|
BlockDecl::~BlockDecl() {
|
|
|
|
}
|
|
|
|
|
|
|
|
void BlockDecl::Destroy(ASTContext& C) {
|
|
|
|
if (Body)
|
|
|
|
Body->Destroy(C);
|
|
|
|
|
|
|
|
for (param_iterator I=param_begin(), E=param_end(); I!=E; ++I)
|
|
|
|
(*I)->Destroy(C);
|
2009-09-09 23:08:12 +08:00
|
|
|
|
|
|
|
C.Deallocate(ParamInfo);
|
2008-10-09 01:01:13 +08:00
|
|
|
Decl::Destroy(C);
|
|
|
|
}
|
2009-03-14 00:56:44 +08:00
|
|
|
|
2010-02-11 09:19:42 +08:00
|
|
|
void BlockDecl::setParams(ParmVarDecl **NewParamInfo,
|
2009-03-14 00:56:44 +08:00
|
|
|
unsigned NParms) {
|
|
|
|
assert(ParamInfo == 0 && "Already has param info!");
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-03-14 00:56:44 +08:00
|
|
|
// Zero params -> null pointer.
|
|
|
|
if (NParms) {
|
|
|
|
NumParams = NParms;
|
2010-02-11 09:19:42 +08:00
|
|
|
void *Mem = getASTContext().Allocate(sizeof(ParmVarDecl*)*NumParams);
|
2009-03-14 00:56:44 +08:00
|
|
|
ParamInfo = new (Mem) ParmVarDecl*[NumParams];
|
|
|
|
memcpy(ParamInfo, NewParamInfo, sizeof(ParmVarDecl*)*NumParams);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
unsigned BlockDecl::getNumParams() const {
|
|
|
|
return NumParams;
|
|
|
|
}
|
2010-01-27 06:01:41 +08:00
|
|
|
|
|
|
|
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
// Other Decl Allocation/Deallocation Method Implementations
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
|
|
|
|
TranslationUnitDecl *TranslationUnitDecl::Create(ASTContext &C) {
|
|
|
|
return new (C) TranslationUnitDecl(C);
|
|
|
|
}
|
|
|
|
|
|
|
|
NamespaceDecl *NamespaceDecl::Create(ASTContext &C, DeclContext *DC,
|
|
|
|
SourceLocation L, IdentifierInfo *Id) {
|
|
|
|
return new (C) NamespaceDecl(DC, L, Id);
|
|
|
|
}
|
|
|
|
|
|
|
|
void NamespaceDecl::Destroy(ASTContext& C) {
|
|
|
|
// NamespaceDecl uses "NextDeclarator" to chain namespace declarations
|
|
|
|
// together. They are all top-level Decls.
|
|
|
|
|
|
|
|
this->~NamespaceDecl();
|
2010-03-15 18:12:16 +08:00
|
|
|
Decl::Destroy(C);
|
2010-01-27 06:01:41 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
ImplicitParamDecl *ImplicitParamDecl::Create(ASTContext &C, DeclContext *DC,
|
|
|
|
SourceLocation L, IdentifierInfo *Id, QualType T) {
|
|
|
|
return new (C) ImplicitParamDecl(ImplicitParam, DC, L, Id, T);
|
|
|
|
}
|
|
|
|
|
|
|
|
FunctionDecl *FunctionDecl::Create(ASTContext &C, DeclContext *DC,
|
|
|
|
SourceLocation L,
|
|
|
|
DeclarationName N, QualType T,
|
|
|
|
TypeSourceInfo *TInfo,
|
|
|
|
StorageClass S, bool isInline,
|
|
|
|
bool hasWrittenPrototype) {
|
|
|
|
FunctionDecl *New
|
|
|
|
= new (C) FunctionDecl(Function, DC, L, N, T, TInfo, S, isInline);
|
|
|
|
New->HasWrittenPrototype = hasWrittenPrototype;
|
|
|
|
return New;
|
|
|
|
}
|
|
|
|
|
|
|
|
BlockDecl *BlockDecl::Create(ASTContext &C, DeclContext *DC, SourceLocation L) {
|
|
|
|
return new (C) BlockDecl(DC, L);
|
|
|
|
}
|
|
|
|
|
|
|
|
EnumConstantDecl *EnumConstantDecl::Create(ASTContext &C, EnumDecl *CD,
|
|
|
|
SourceLocation L,
|
|
|
|
IdentifierInfo *Id, QualType T,
|
|
|
|
Expr *E, const llvm::APSInt &V) {
|
|
|
|
return new (C) EnumConstantDecl(CD, L, Id, T, E, V);
|
|
|
|
}
|
|
|
|
|
|
|
|
void EnumConstantDecl::Destroy(ASTContext& C) {
|
|
|
|
if (Init) Init->Destroy(C);
|
2010-03-15 18:12:16 +08:00
|
|
|
ValueDecl::Destroy(C);
|
2010-01-27 06:01:41 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
TypedefDecl *TypedefDecl::Create(ASTContext &C, DeclContext *DC,
|
|
|
|
SourceLocation L, IdentifierInfo *Id,
|
|
|
|
TypeSourceInfo *TInfo) {
|
|
|
|
return new (C) TypedefDecl(DC, L, Id, TInfo);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Anchor TypedefDecl's vtable here.
|
|
|
|
TypedefDecl::~TypedefDecl() {}
|
|
|
|
|
|
|
|
FileScopeAsmDecl *FileScopeAsmDecl::Create(ASTContext &C, DeclContext *DC,
|
|
|
|
SourceLocation L,
|
|
|
|
StringLiteral *Str) {
|
|
|
|
return new (C) FileScopeAsmDecl(DC, L, Str);
|
|
|
|
}
|