2008-04-08 13:04:30 +08:00
|
|
|
|
//===------ SemaDeclCXX.cpp - Semantic Analysis for C++ Declarations ------===//
|
|
|
|
|
//
|
|
|
|
|
// The LLVM Compiler Infrastructure
|
|
|
|
|
//
|
|
|
|
|
// This file is distributed under the University of Illinois Open Source
|
|
|
|
|
// License. See LICENSE.TXT for details.
|
|
|
|
|
//
|
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
|
//
|
|
|
|
|
// This file implements semantic analysis for C++ declarations.
|
|
|
|
|
//
|
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
|
|
|
|
|
|
#include "Sema.h"
|
2008-11-05 12:29:56 +08:00
|
|
|
|
#include "SemaInherit.h"
|
2008-08-09 08:58:37 +08:00
|
|
|
|
#include "clang/AST/ASTConsumer.h"
|
2008-04-14 05:30:24 +08:00
|
|
|
|
#include "clang/AST/ASTContext.h"
|
2009-03-24 09:19:16 +08:00
|
|
|
|
#include "clang/AST/DeclVisitor.h"
|
2008-10-23 05:13:31 +08:00
|
|
|
|
#include "clang/AST/TypeOrdering.h"
|
2008-04-10 10:22:51 +08:00
|
|
|
|
#include "clang/AST/StmtVisitor.h"
|
2008-10-07 02:37:09 +08:00
|
|
|
|
#include "clang/Lex/Preprocessor.h"
|
2008-08-11 11:27:53 +08:00
|
|
|
|
#include "clang/Parse/DeclSpec.h"
|
2008-12-23 08:26:44 +08:00
|
|
|
|
#include "llvm/ADT/STLExtras.h"
|
2008-04-10 10:22:51 +08:00
|
|
|
|
#include "llvm/Support/Compiler.h"
|
2008-10-22 00:13:35 +08:00
|
|
|
|
#include <algorithm> // for std::equal
|
2008-10-23 01:49:05 +08:00
|
|
|
|
#include <map>
|
2008-04-08 13:04:30 +08:00
|
|
|
|
|
|
|
|
|
using namespace clang;
|
|
|
|
|
|
2008-04-10 10:22:51 +08:00
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
|
// CheckDefaultArgumentVisitor
|
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
|
|
2008-04-13 07:52:44 +08:00
|
|
|
|
namespace {
|
|
|
|
|
/// CheckDefaultArgumentVisitor - C++ [dcl.fct.default] Traverses
|
|
|
|
|
/// the default argument of a parameter to determine whether it
|
|
|
|
|
/// contains any ill-formed subexpressions. For example, this will
|
|
|
|
|
/// diagnose the use of local variables or parameters within the
|
|
|
|
|
/// default argument expression.
|
|
|
|
|
class VISIBILITY_HIDDEN CheckDefaultArgumentVisitor
|
2008-07-27 06:17:49 +08:00
|
|
|
|
: public StmtVisitor<CheckDefaultArgumentVisitor, bool> {
|
2008-04-13 07:52:44 +08:00
|
|
|
|
Expr *DefaultArg;
|
|
|
|
|
Sema *S;
|
|
|
|
|
|
|
|
|
|
public:
|
|
|
|
|
CheckDefaultArgumentVisitor(Expr *defarg, Sema *s)
|
|
|
|
|
: DefaultArg(defarg), S(s) {}
|
|
|
|
|
|
|
|
|
|
bool VisitExpr(Expr *Node);
|
|
|
|
|
bool VisitDeclRefExpr(DeclRefExpr *DRE);
|
2008-11-04 22:32:21 +08:00
|
|
|
|
bool VisitCXXThisExpr(CXXThisExpr *ThisE);
|
2008-04-13 07:52:44 +08:00
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
/// VisitExpr - Visit all of the children of this expression.
|
|
|
|
|
bool CheckDefaultArgumentVisitor::VisitExpr(Expr *Node) {
|
|
|
|
|
bool IsInvalid = false;
|
2008-07-27 06:17:49 +08:00
|
|
|
|
for (Stmt::child_iterator I = Node->child_begin(),
|
|
|
|
|
E = Node->child_end(); I != E; ++I)
|
|
|
|
|
IsInvalid |= Visit(*I);
|
2008-04-13 07:52:44 +08:00
|
|
|
|
return IsInvalid;
|
2008-04-10 10:22:51 +08:00
|
|
|
|
}
|
|
|
|
|
|
2008-04-13 07:52:44 +08:00
|
|
|
|
/// VisitDeclRefExpr - Visit a reference to a declaration, to
|
|
|
|
|
/// determine whether this declaration can be used in the default
|
|
|
|
|
/// argument expression.
|
|
|
|
|
bool CheckDefaultArgumentVisitor::VisitDeclRefExpr(DeclRefExpr *DRE) {
|
2008-10-22 00:13:35 +08:00
|
|
|
|
NamedDecl *Decl = DRE->getDecl();
|
2008-04-13 07:52:44 +08:00
|
|
|
|
if (ParmVarDecl *Param = dyn_cast<ParmVarDecl>(Decl)) {
|
|
|
|
|
// C++ [dcl.fct.default]p9
|
|
|
|
|
// Default arguments are evaluated each time the function is
|
|
|
|
|
// called. The order of evaluation of function arguments is
|
|
|
|
|
// unspecified. Consequently, parameters of a function shall not
|
|
|
|
|
// be used in default argument expressions, even if they are not
|
|
|
|
|
// evaluated. Parameters of a function declared before a default
|
|
|
|
|
// argument expression are in scope and can hide namespace and
|
|
|
|
|
// class member names.
|
|
|
|
|
return S->Diag(DRE->getSourceRange().getBegin(),
|
2008-11-19 13:08:23 +08:00
|
|
|
|
diag::err_param_default_argument_references_param)
|
2008-11-24 05:45:46 +08:00
|
|
|
|
<< Param->getDeclName() << DefaultArg->getSourceRange();
|
2008-04-16 06:42:06 +08:00
|
|
|
|
} else if (VarDecl *VDecl = dyn_cast<VarDecl>(Decl)) {
|
2008-04-13 07:52:44 +08:00
|
|
|
|
// C++ [dcl.fct.default]p7
|
|
|
|
|
// Local variables shall not be used in default argument
|
|
|
|
|
// expressions.
|
2008-04-16 06:42:06 +08:00
|
|
|
|
if (VDecl->isBlockVarDecl())
|
|
|
|
|
return S->Diag(DRE->getSourceRange().getBegin(),
|
2008-11-19 13:08:23 +08:00
|
|
|
|
diag::err_param_default_argument_references_local)
|
2008-11-24 05:45:46 +08:00
|
|
|
|
<< VDecl->getDeclName() << DefaultArg->getSourceRange();
|
2008-04-13 07:52:44 +08:00
|
|
|
|
}
|
|
|
|
|
|
2008-11-04 21:41:56 +08:00
|
|
|
|
return false;
|
|
|
|
|
}
|
2008-04-13 07:52:44 +08:00
|
|
|
|
|
2008-11-04 22:32:21 +08:00
|
|
|
|
/// VisitCXXThisExpr - Visit a C++ "this" expression.
|
|
|
|
|
bool CheckDefaultArgumentVisitor::VisitCXXThisExpr(CXXThisExpr *ThisE) {
|
|
|
|
|
// C++ [dcl.fct.default]p8:
|
|
|
|
|
// The keyword this shall not be used in a default argument of a
|
|
|
|
|
// member function.
|
|
|
|
|
return S->Diag(ThisE->getSourceRange().getBegin(),
|
2008-11-19 13:08:23 +08:00
|
|
|
|
diag::err_param_default_argument_references_this)
|
|
|
|
|
<< ThisE->getSourceRange();
|
2008-04-13 07:52:44 +08:00
|
|
|
|
}
|
2008-04-10 10:22:51 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/// ActOnParamDefaultArgument - Check whether the default argument
|
|
|
|
|
/// provided for a function parameter is well-formed. If so, attach it
|
|
|
|
|
/// to the parameter declaration.
|
2008-04-08 13:04:30 +08:00
|
|
|
|
void
|
2009-03-29 03:18:32 +08:00
|
|
|
|
Sema::ActOnParamDefaultArgument(DeclPtrTy param, SourceLocation EqualLoc,
|
2009-03-16 01:47:39 +08:00
|
|
|
|
ExprArg defarg) {
|
2009-03-29 03:18:32 +08:00
|
|
|
|
ParmVarDecl *Param = cast<ParmVarDecl>(param.getAs<Decl>());
|
2009-06-13 00:51:40 +08:00
|
|
|
|
UnparsedDefaultArgLocs.erase(Param);
|
|
|
|
|
|
2009-05-02 03:30:39 +08:00
|
|
|
|
ExprOwningPtr<Expr> DefaultArg(this, defarg.takeAs<Expr>());
|
2008-04-08 13:04:30 +08:00
|
|
|
|
QualType ParamType = Param->getType();
|
|
|
|
|
|
|
|
|
|
// Default arguments are only permitted in C++
|
|
|
|
|
if (!getLangOptions().CPlusPlus) {
|
2008-11-19 13:08:23 +08:00
|
|
|
|
Diag(EqualLoc, diag::err_param_default_argument)
|
|
|
|
|
<< DefaultArg->getSourceRange();
|
2008-12-17 05:30:33 +08:00
|
|
|
|
Param->setInvalidDecl();
|
2008-04-08 13:04:30 +08:00
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// C++ [dcl.fct.default]p5
|
|
|
|
|
// A default argument expression is implicitly converted (clause
|
|
|
|
|
// 4) to the parameter type. The default argument expression has
|
|
|
|
|
// the same semantic constraints as the initializer expression in
|
|
|
|
|
// a declaration of a variable of the parameter type, using the
|
|
|
|
|
// copy-initialization semantics (8.5).
|
|
|
|
|
Expr *DefaultArgPtr = DefaultArg.get();
|
2008-12-24 08:01:03 +08:00
|
|
|
|
bool DefaultInitFailed = CheckInitializerTypes(DefaultArgPtr, ParamType,
|
|
|
|
|
EqualLoc,
|
2009-01-14 23:45:31 +08:00
|
|
|
|
Param->getDeclName(),
|
|
|
|
|
/*DirectInit=*/false);
|
2008-04-08 13:04:30 +08:00
|
|
|
|
if (DefaultArgPtr != DefaultArg.get()) {
|
|
|
|
|
DefaultArg.take();
|
|
|
|
|
DefaultArg.reset(DefaultArgPtr);
|
|
|
|
|
}
|
2008-11-04 21:57:51 +08:00
|
|
|
|
if (DefaultInitFailed) {
|
2008-04-08 13:04:30 +08:00
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
2008-04-10 10:22:51 +08:00
|
|
|
|
// Check that the default argument is well-formed
|
2008-04-13 07:52:44 +08:00
|
|
|
|
CheckDefaultArgumentVisitor DefaultArgChecker(DefaultArg.get(), this);
|
2008-12-17 05:30:33 +08:00
|
|
|
|
if (DefaultArgChecker.Visit(DefaultArg.get())) {
|
|
|
|
|
Param->setInvalidDecl();
|
2008-04-10 10:22:51 +08:00
|
|
|
|
return;
|
2008-12-17 05:30:33 +08:00
|
|
|
|
}
|
2008-04-10 10:22:51 +08:00
|
|
|
|
|
2009-06-16 11:37:31 +08:00
|
|
|
|
DefaultArgPtr = MaybeCreateCXXExprWithTemporaries(DefaultArg.take(),
|
|
|
|
|
/*DestroyTemps=*/false);
|
|
|
|
|
|
2008-04-08 13:04:30 +08:00
|
|
|
|
// Okay: add the default argument to the parameter
|
2009-06-16 11:37:31 +08:00
|
|
|
|
Param->setDefaultArg(DefaultArgPtr);
|
2008-04-08 13:04:30 +08:00
|
|
|
|
}
|
|
|
|
|
|
2008-12-24 08:01:03 +08:00
|
|
|
|
/// ActOnParamUnparsedDefaultArgument - We've seen a default
|
|
|
|
|
/// argument for a function parameter, but we can't parse it yet
|
|
|
|
|
/// because we're inside a class definition. Note that this default
|
|
|
|
|
/// argument will be parsed later.
|
2009-03-29 03:18:32 +08:00
|
|
|
|
void Sema::ActOnParamUnparsedDefaultArgument(DeclPtrTy param,
|
2009-06-13 00:51:40 +08:00
|
|
|
|
SourceLocation EqualLoc,
|
|
|
|
|
SourceLocation ArgLoc) {
|
2009-03-29 03:18:32 +08:00
|
|
|
|
ParmVarDecl *Param = cast<ParmVarDecl>(param.getAs<Decl>());
|
2008-12-24 08:01:03 +08:00
|
|
|
|
if (Param)
|
|
|
|
|
Param->setUnparsedDefaultArg();
|
2009-06-13 00:51:40 +08:00
|
|
|
|
|
|
|
|
|
UnparsedDefaultArgLocs[Param] = ArgLoc;
|
2008-12-24 08:01:03 +08:00
|
|
|
|
}
|
|
|
|
|
|
2008-12-17 05:30:33 +08:00
|
|
|
|
/// ActOnParamDefaultArgumentError - Parsing or semantic analysis of
|
|
|
|
|
/// the default argument for the parameter param failed.
|
2009-03-29 03:18:32 +08:00
|
|
|
|
void Sema::ActOnParamDefaultArgumentError(DeclPtrTy param) {
|
2009-06-13 00:51:40 +08:00
|
|
|
|
ParmVarDecl *Param = cast<ParmVarDecl>(param.getAs<Decl>());
|
|
|
|
|
|
|
|
|
|
Param->setInvalidDecl();
|
|
|
|
|
|
|
|
|
|
UnparsedDefaultArgLocs.erase(Param);
|
2008-12-17 05:30:33 +08:00
|
|
|
|
}
|
|
|
|
|
|
2008-05-07 12:49:29 +08:00
|
|
|
|
/// CheckExtraCXXDefaultArguments - Check for any extra default
|
|
|
|
|
/// arguments in the declarator, which is not a function declaration
|
|
|
|
|
/// or definition and therefore is not permitted to have default
|
|
|
|
|
/// arguments. This routine should be invoked for every declarator
|
|
|
|
|
/// that is not a function declaration or definition.
|
|
|
|
|
void Sema::CheckExtraCXXDefaultArguments(Declarator &D) {
|
|
|
|
|
// C++ [dcl.fct.default]p3
|
|
|
|
|
// A default argument expression shall be specified only in the
|
|
|
|
|
// parameter-declaration-clause of a function declaration or in a
|
|
|
|
|
// template-parameter (14.1). It shall not be specified for a
|
|
|
|
|
// parameter pack. If it is specified in a
|
|
|
|
|
// parameter-declaration-clause, it shall not occur within a
|
|
|
|
|
// declarator or abstract-declarator of a parameter-declaration.
|
2009-03-29 03:18:32 +08:00
|
|
|
|
for (unsigned i = 0, e = D.getNumTypeObjects(); i != e; ++i) {
|
2008-05-07 12:49:29 +08:00
|
|
|
|
DeclaratorChunk &chunk = D.getTypeObject(i);
|
|
|
|
|
if (chunk.Kind == DeclaratorChunk::Function) {
|
2009-03-29 03:18:32 +08:00
|
|
|
|
for (unsigned argIdx = 0, e = chunk.Fun.NumArgs; argIdx != e; ++argIdx) {
|
|
|
|
|
ParmVarDecl *Param =
|
|
|
|
|
cast<ParmVarDecl>(chunk.Fun.ArgInfo[argIdx].Param.getAs<Decl>());
|
2008-12-24 08:01:03 +08:00
|
|
|
|
if (Param->hasUnparsedDefaultArg()) {
|
|
|
|
|
CachedTokens *Toks = chunk.Fun.ArgInfo[argIdx].DefaultArgTokens;
|
2008-12-17 05:30:33 +08:00
|
|
|
|
Diag(Param->getLocation(), diag::err_param_default_argument_nonfunc)
|
|
|
|
|
<< SourceRange((*Toks)[1].getLocation(), Toks->back().getLocation());
|
|
|
|
|
delete Toks;
|
|
|
|
|
chunk.Fun.ArgInfo[argIdx].DefaultArgTokens = 0;
|
2008-12-24 08:01:03 +08:00
|
|
|
|
} else if (Param->getDefaultArg()) {
|
|
|
|
|
Diag(Param->getLocation(), diag::err_param_default_argument_nonfunc)
|
|
|
|
|
<< Param->getDefaultArg()->getSourceRange();
|
|
|
|
|
Param->setDefaultArg(0);
|
2008-05-07 12:49:29 +08:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2008-04-08 13:04:30 +08:00
|
|
|
|
// MergeCXXFunctionDecl - Merge two declarations of the same C++
|
|
|
|
|
// function, once we already know that they have the same
|
2009-02-17 01:45:42 +08:00
|
|
|
|
// type. Subroutine of MergeFunctionDecl. Returns true if there was an
|
|
|
|
|
// error, false otherwise.
|
|
|
|
|
bool Sema::MergeCXXFunctionDecl(FunctionDecl *New, FunctionDecl *Old) {
|
|
|
|
|
bool Invalid = false;
|
|
|
|
|
|
2008-04-08 13:04:30 +08:00
|
|
|
|
// C++ [dcl.fct.default]p4:
|
|
|
|
|
//
|
|
|
|
|
// For non-template functions, default arguments can be added in
|
|
|
|
|
// later declarations of a function in the same
|
|
|
|
|
// scope. Declarations in different scopes have completely
|
|
|
|
|
// distinct sets of default arguments. That is, declarations in
|
|
|
|
|
// inner scopes do not acquire default arguments from
|
|
|
|
|
// declarations in outer scopes, and vice versa. In a given
|
|
|
|
|
// function declaration, all parameters subsequent to a
|
|
|
|
|
// parameter with a default argument shall have default
|
|
|
|
|
// arguments supplied in this or previous declarations. A
|
|
|
|
|
// default argument shall not be redefined by a later
|
|
|
|
|
// declaration (not even to the same value).
|
|
|
|
|
for (unsigned p = 0, NumParams = Old->getNumParams(); p < NumParams; ++p) {
|
|
|
|
|
ParmVarDecl *OldParam = Old->getParamDecl(p);
|
|
|
|
|
ParmVarDecl *NewParam = New->getParamDecl(p);
|
|
|
|
|
|
|
|
|
|
if(OldParam->getDefaultArg() && NewParam->getDefaultArg()) {
|
|
|
|
|
Diag(NewParam->getLocation(),
|
2008-11-19 13:08:23 +08:00
|
|
|
|
diag::err_param_default_argument_redefinition)
|
|
|
|
|
<< NewParam->getDefaultArg()->getSourceRange();
|
2008-11-24 07:12:31 +08:00
|
|
|
|
Diag(OldParam->getLocation(), diag::note_previous_definition);
|
2009-02-17 01:45:42 +08:00
|
|
|
|
Invalid = true;
|
2008-04-08 13:04:30 +08:00
|
|
|
|
} else if (OldParam->getDefaultArg()) {
|
|
|
|
|
// Merge the old default argument into the new parameter
|
|
|
|
|
NewParam->setDefaultArg(OldParam->getDefaultArg());
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2009-02-17 01:45:42 +08:00
|
|
|
|
return Invalid;
|
2008-04-08 13:04:30 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/// CheckCXXDefaultArguments - Verify that the default arguments for a
|
|
|
|
|
/// function declaration are well-formed according to C++
|
|
|
|
|
/// [dcl.fct.default].
|
|
|
|
|
void Sema::CheckCXXDefaultArguments(FunctionDecl *FD) {
|
|
|
|
|
unsigned NumParams = FD->getNumParams();
|
|
|
|
|
unsigned p;
|
|
|
|
|
|
|
|
|
|
// Find first parameter with a default argument
|
|
|
|
|
for (p = 0; p < NumParams; ++p) {
|
|
|
|
|
ParmVarDecl *Param = FD->getParamDecl(p);
|
|
|
|
|
if (Param->getDefaultArg())
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// C++ [dcl.fct.default]p4:
|
|
|
|
|
// In a given function declaration, all parameters
|
|
|
|
|
// subsequent to a parameter with a default argument shall
|
|
|
|
|
// have default arguments supplied in this or previous
|
|
|
|
|
// declarations. A default argument shall not be redefined
|
|
|
|
|
// by a later declaration (not even to the same value).
|
|
|
|
|
unsigned LastMissingDefaultArg = 0;
|
|
|
|
|
for(; p < NumParams; ++p) {
|
|
|
|
|
ParmVarDecl *Param = FD->getParamDecl(p);
|
|
|
|
|
if (!Param->getDefaultArg()) {
|
2008-12-17 05:30:33 +08:00
|
|
|
|
if (Param->isInvalidDecl())
|
|
|
|
|
/* We already complained about this parameter. */;
|
|
|
|
|
else if (Param->getIdentifier())
|
2008-04-08 13:04:30 +08:00
|
|
|
|
Diag(Param->getLocation(),
|
2008-11-19 13:08:23 +08:00
|
|
|
|
diag::err_param_default_argument_missing_name)
|
2008-11-19 15:32:16 +08:00
|
|
|
|
<< Param->getIdentifier();
|
2008-04-08 13:04:30 +08:00
|
|
|
|
else
|
|
|
|
|
Diag(Param->getLocation(),
|
|
|
|
|
diag::err_param_default_argument_missing);
|
|
|
|
|
|
|
|
|
|
LastMissingDefaultArg = p;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (LastMissingDefaultArg > 0) {
|
|
|
|
|
// Some default arguments were missing. Clear out all of the
|
|
|
|
|
// default arguments up to (and including) the last missing
|
|
|
|
|
// default argument, so that we leave the function parameters
|
|
|
|
|
// in a semantically valid state.
|
|
|
|
|
for (p = 0; p <= LastMissingDefaultArg; ++p) {
|
|
|
|
|
ParmVarDecl *Param = FD->getParamDecl(p);
|
2009-06-13 00:51:40 +08:00
|
|
|
|
if (Param->hasDefaultArg()) {
|
2008-12-24 08:01:03 +08:00
|
|
|
|
if (!Param->hasUnparsedDefaultArg())
|
|
|
|
|
Param->getDefaultArg()->Destroy(Context);
|
2008-04-08 13:04:30 +08:00
|
|
|
|
Param->setDefaultArg(0);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
2008-04-14 05:30:24 +08:00
|
|
|
|
|
2008-10-31 17:07:45 +08:00
|
|
|
|
/// isCurrentClassName - Determine whether the identifier II is the
|
|
|
|
|
/// name of the class type currently being defined. In the case of
|
|
|
|
|
/// nested classes, this will only return true if II is the name of
|
|
|
|
|
/// the innermost class.
|
2008-11-09 00:45:02 +08:00
|
|
|
|
bool Sema::isCurrentClassName(const IdentifierInfo &II, Scope *,
|
|
|
|
|
const CXXScopeSpec *SS) {
|
2008-11-09 01:17:31 +08:00
|
|
|
|
CXXRecordDecl *CurDecl;
|
Introduce a representation for types that we referred to via a
qualified name, e.g.,
foo::x
so that we retain the nested-name-specifier as written in the source
code and can reproduce that qualified name when printing the types
back (e.g., in diagnostics). This is PR3493, which won't be complete
until finished the other tasks mentioned near the end of this commit.
The parser's representation of nested-name-specifiers, CXXScopeSpec,
is now a bit fatter, because it needs to contain the scopes that
precede each '::' and keep track of whether the global scoping
operator '::' was at the beginning. For example, we need to keep track
of the leading '::', 'foo', and 'bar' in
::foo::bar::x
The Action's CXXScopeTy * is no longer a DeclContext *. It's now the
opaque version of the new NestedNameSpecifier, which contains a single
component of a nested-name-specifier (either a DeclContext * or a Type
*, bitmangled).
The new sugar type QualifiedNameType composes a sequence of
NestedNameSpecifiers with a representation of the type we're actually
referring to. At present, we only build QualifiedNameType nodes within
Sema::getTypeName. This will be extended to other type-constructing
actions (e.g., ActOnClassTemplateId).
Also on the way: QualifiedDeclRefExprs will also store a sequence of
NestedNameSpecifiers, so that we can print out the property
nested-name-specifier. I expect to also use this for handling
dependent names like Fibonacci<I - 1>::value.
llvm-svn: 67265
2009-03-19 08:18:19 +08:00
|
|
|
|
if (SS && SS->isSet() && !SS->isInvalid()) {
|
|
|
|
|
DeclContext *DC = computeDeclContext(*SS);
|
2008-11-09 01:17:31 +08:00
|
|
|
|
CurDecl = dyn_cast_or_null<CXXRecordDecl>(DC);
|
|
|
|
|
} else
|
|
|
|
|
CurDecl = dyn_cast_or_null<CXXRecordDecl>(CurContext);
|
|
|
|
|
|
|
|
|
|
if (CurDecl)
|
2008-10-31 17:07:45 +08:00
|
|
|
|
return &II == CurDecl->getIdentifier();
|
|
|
|
|
else
|
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
|
2009-03-03 12:44:36 +08:00
|
|
|
|
/// \brief Check the validity of a C++ base class specifier.
|
|
|
|
|
///
|
|
|
|
|
/// \returns a new CXXBaseSpecifier if well-formed, emits diagnostics
|
|
|
|
|
/// and returns NULL otherwise.
|
|
|
|
|
CXXBaseSpecifier *
|
|
|
|
|
Sema::CheckBaseSpecifier(CXXRecordDecl *Class,
|
|
|
|
|
SourceRange SpecifierRange,
|
2008-10-23 01:49:05 +08:00
|
|
|
|
bool Virtual, AccessSpecifier Access,
|
2009-03-03 12:44:36 +08:00
|
|
|
|
QualType BaseType,
|
|
|
|
|
SourceLocation BaseLoc) {
|
|
|
|
|
// C++ [class.union]p1:
|
|
|
|
|
// A union shall not have base classes.
|
|
|
|
|
if (Class->isUnion()) {
|
|
|
|
|
Diag(Class->getLocation(), diag::err_base_clause_on_union)
|
|
|
|
|
<< SpecifierRange;
|
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (BaseType->isDependentType())
|
|
|
|
|
return new CXXBaseSpecifier(SpecifierRange, Virtual,
|
|
|
|
|
Class->getTagKind() == RecordDecl::TK_class,
|
|
|
|
|
Access, BaseType);
|
2008-04-14 05:30:24 +08:00
|
|
|
|
|
|
|
|
|
// Base specifiers must be record types.
|
2009-03-03 12:44:36 +08:00
|
|
|
|
if (!BaseType->isRecordType()) {
|
|
|
|
|
Diag(BaseLoc, diag::err_base_must_be_class) << SpecifierRange;
|
|
|
|
|
return 0;
|
|
|
|
|
}
|
2008-04-14 05:30:24 +08:00
|
|
|
|
|
|
|
|
|
// C++ [class.union]p1:
|
|
|
|
|
// A union shall not be used as a base class.
|
2009-03-03 12:44:36 +08:00
|
|
|
|
if (BaseType->isUnionType()) {
|
|
|
|
|
Diag(BaseLoc, diag::err_union_as_base_class) << SpecifierRange;
|
|
|
|
|
return 0;
|
|
|
|
|
}
|
2008-04-14 05:30:24 +08:00
|
|
|
|
|
|
|
|
|
// C++ [class.derived]p2:
|
|
|
|
|
// The class-name in a base-specifier shall not be an incompletely
|
|
|
|
|
// defined class.
|
2009-03-10 00:13:40 +08:00
|
|
|
|
if (RequireCompleteType(BaseLoc, BaseType, diag::err_incomplete_base_class,
|
2009-03-10 08:06:19 +08:00
|
|
|
|
SpecifierRange))
|
2009-03-03 12:44:36 +08:00
|
|
|
|
return 0;
|
2008-04-14 05:30:24 +08:00
|
|
|
|
|
2008-11-06 23:59:35 +08:00
|
|
|
|
// If the base class is polymorphic, the new one is, too.
|
|
|
|
|
RecordDecl *BaseDecl = BaseType->getAsRecordType()->getDecl();
|
|
|
|
|
assert(BaseDecl && "Record type has no declaration");
|
|
|
|
|
BaseDecl = BaseDecl->getDefinition(Context);
|
|
|
|
|
assert(BaseDecl && "Base type is not incomplete, but has no definition");
|
2008-11-19 13:08:23 +08:00
|
|
|
|
if (cast<CXXRecordDecl>(BaseDecl)->isPolymorphic())
|
2009-03-03 12:44:36 +08:00
|
|
|
|
Class->setPolymorphic(true);
|
2009-01-06 04:52:13 +08:00
|
|
|
|
|
|
|
|
|
// C++ [dcl.init.aggr]p1:
|
|
|
|
|
// An aggregate is [...] a class with [...] no base classes [...].
|
2009-03-03 12:44:36 +08:00
|
|
|
|
Class->setAggregate(false);
|
|
|
|
|
Class->setPOD(false);
|
2008-11-06 23:59:35 +08:00
|
|
|
|
|
2009-04-16 08:08:20 +08:00
|
|
|
|
if (Virtual) {
|
|
|
|
|
// C++ [class.ctor]p5:
|
|
|
|
|
// A constructor is trivial if its class has no virtual base classes.
|
|
|
|
|
Class->setHasTrivialConstructor(false);
|
|
|
|
|
} else {
|
|
|
|
|
// C++ [class.ctor]p5:
|
|
|
|
|
// A constructor is trivial if all the direct base classes of its
|
|
|
|
|
// class have trivial constructors.
|
|
|
|
|
Class->setHasTrivialConstructor(cast<CXXRecordDecl>(BaseDecl)->
|
|
|
|
|
hasTrivialConstructor());
|
|
|
|
|
}
|
2009-04-17 10:34:54 +08:00
|
|
|
|
|
|
|
|
|
// C++ [class.ctor]p3:
|
|
|
|
|
// A destructor is trivial if all the direct base classes of its class
|
|
|
|
|
// have trivial destructors.
|
|
|
|
|
Class->setHasTrivialDestructor(cast<CXXRecordDecl>(BaseDecl)->
|
|
|
|
|
hasTrivialDestructor());
|
2009-04-16 08:08:20 +08:00
|
|
|
|
|
2008-10-23 01:49:05 +08:00
|
|
|
|
// Create the base specifier.
|
2009-03-03 12:44:36 +08:00
|
|
|
|
// FIXME: Allocate via ASTContext?
|
2008-10-24 02:13:27 +08:00
|
|
|
|
return new CXXBaseSpecifier(SpecifierRange, Virtual,
|
2009-03-03 12:44:36 +08:00
|
|
|
|
Class->getTagKind() == RecordDecl::TK_class,
|
|
|
|
|
Access, BaseType);
|
2008-10-23 01:49:05 +08:00
|
|
|
|
}
|
|
|
|
|
|
2009-03-03 12:44:36 +08:00
|
|
|
|
/// ActOnBaseSpecifier - Parsed a base specifier. A base specifier is
|
|
|
|
|
/// one entry in the base class list of a class specifier, for
|
|
|
|
|
/// example:
|
|
|
|
|
/// class foo : public bar, virtual private baz {
|
|
|
|
|
/// 'public bar' and 'virtual private baz' are each base-specifiers.
|
|
|
|
|
Sema::BaseResult
|
2009-03-29 03:18:32 +08:00
|
|
|
|
Sema::ActOnBaseSpecifier(DeclPtrTy classdecl, SourceRange SpecifierRange,
|
2009-03-03 12:44:36 +08:00
|
|
|
|
bool Virtual, AccessSpecifier Access,
|
|
|
|
|
TypeTy *basetype, SourceLocation BaseLoc) {
|
2009-03-10 07:48:35 +08:00
|
|
|
|
AdjustDeclIfTemplate(classdecl);
|
2009-03-29 03:18:32 +08:00
|
|
|
|
CXXRecordDecl *Class = cast<CXXRecordDecl>(classdecl.getAs<Decl>());
|
2009-03-03 12:44:36 +08:00
|
|
|
|
QualType BaseType = QualType::getFromOpaquePtr(basetype);
|
|
|
|
|
if (CXXBaseSpecifier *BaseSpec = CheckBaseSpecifier(Class, SpecifierRange,
|
|
|
|
|
Virtual, Access,
|
|
|
|
|
BaseType, BaseLoc))
|
|
|
|
|
return BaseSpec;
|
|
|
|
|
|
|
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/// \brief Performs the actual work of attaching the given base class
|
|
|
|
|
/// specifiers to a C++ class.
|
|
|
|
|
bool Sema::AttachBaseSpecifiers(CXXRecordDecl *Class, CXXBaseSpecifier **Bases,
|
|
|
|
|
unsigned NumBases) {
|
|
|
|
|
if (NumBases == 0)
|
|
|
|
|
return false;
|
2008-10-23 01:49:05 +08:00
|
|
|
|
|
|
|
|
|
// Used to keep track of which base types we have already seen, so
|
|
|
|
|
// that we can properly diagnose redundant direct base types. Note
|
2008-10-24 02:13:27 +08:00
|
|
|
|
// that the key is always the unqualified canonical type of the base
|
|
|
|
|
// class.
|
2008-10-23 01:49:05 +08:00
|
|
|
|
std::map<QualType, CXXBaseSpecifier*, QualTypeOrdering> KnownBaseTypes;
|
|
|
|
|
|
|
|
|
|
// Copy non-redundant base specifiers into permanent storage.
|
2008-10-24 02:13:27 +08:00
|
|
|
|
unsigned NumGoodBases = 0;
|
2009-03-03 12:44:36 +08:00
|
|
|
|
bool Invalid = false;
|
2008-10-24 02:13:27 +08:00
|
|
|
|
for (unsigned idx = 0; idx < NumBases; ++idx) {
|
2008-10-23 01:49:05 +08:00
|
|
|
|
QualType NewBaseType
|
2009-03-03 12:44:36 +08:00
|
|
|
|
= Context.getCanonicalType(Bases[idx]->getType());
|
2008-10-24 02:13:27 +08:00
|
|
|
|
NewBaseType = NewBaseType.getUnqualifiedType();
|
|
|
|
|
|
2008-10-23 01:49:05 +08:00
|
|
|
|
if (KnownBaseTypes[NewBaseType]) {
|
|
|
|
|
// C++ [class.mi]p3:
|
|
|
|
|
// A class shall not be specified as a direct base class of a
|
|
|
|
|
// derived class more than once.
|
2009-03-03 12:44:36 +08:00
|
|
|
|
Diag(Bases[idx]->getSourceRange().getBegin(),
|
2008-11-19 13:08:23 +08:00
|
|
|
|
diag::err_duplicate_base_class)
|
2008-11-24 14:25:27 +08:00
|
|
|
|
<< KnownBaseTypes[NewBaseType]->getType()
|
2009-03-03 12:44:36 +08:00
|
|
|
|
<< Bases[idx]->getSourceRange();
|
2008-10-24 02:13:27 +08:00
|
|
|
|
|
|
|
|
|
// Delete the duplicate base class specifier; we're going to
|
|
|
|
|
// overwrite its pointer later.
|
2009-03-03 12:44:36 +08:00
|
|
|
|
delete Bases[idx];
|
|
|
|
|
|
|
|
|
|
Invalid = true;
|
2008-10-23 01:49:05 +08:00
|
|
|
|
} else {
|
|
|
|
|
// Okay, add this new base class.
|
2009-03-03 12:44:36 +08:00
|
|
|
|
KnownBaseTypes[NewBaseType] = Bases[idx];
|
|
|
|
|
Bases[NumGoodBases++] = Bases[idx];
|
2008-10-23 01:49:05 +08:00
|
|
|
|
}
|
|
|
|
|
}
|
2008-04-14 05:30:24 +08:00
|
|
|
|
|
2008-10-23 01:49:05 +08:00
|
|
|
|
// Attach the remaining base class specifiers to the derived class.
|
2009-03-03 12:44:36 +08:00
|
|
|
|
Class->setBases(Bases, NumGoodBases);
|
2008-10-24 02:13:27 +08:00
|
|
|
|
|
|
|
|
|
// Delete the remaining (good) base class specifiers, since their
|
|
|
|
|
// data has been copied into the CXXRecordDecl.
|
|
|
|
|
for (unsigned idx = 0; idx < NumGoodBases; ++idx)
|
2009-03-03 12:44:36 +08:00
|
|
|
|
delete Bases[idx];
|
|
|
|
|
|
|
|
|
|
return Invalid;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/// ActOnBaseSpecifiers - Attach the given base specifiers to the
|
|
|
|
|
/// class, after checking whether there are any duplicate base
|
|
|
|
|
/// classes.
|
2009-03-29 03:18:32 +08:00
|
|
|
|
void Sema::ActOnBaseSpecifiers(DeclPtrTy ClassDecl, BaseTy **Bases,
|
2009-03-03 12:44:36 +08:00
|
|
|
|
unsigned NumBases) {
|
|
|
|
|
if (!ClassDecl || !Bases || !NumBases)
|
|
|
|
|
return;
|
|
|
|
|
|
|
|
|
|
AdjustDeclIfTemplate(ClassDecl);
|
2009-03-29 03:18:32 +08:00
|
|
|
|
AttachBaseSpecifiers(cast<CXXRecordDecl>(ClassDecl.getAs<Decl>()),
|
2009-03-03 12:44:36 +08:00
|
|
|
|
(CXXBaseSpecifier**)(Bases), NumBases);
|
2008-04-14 05:30:24 +08:00
|
|
|
|
}
|
2008-04-27 21:50:30 +08:00
|
|
|
|
|
2008-07-01 18:37:29 +08:00
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
|
// C++ class member Handling
|
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
|
|
|
|
|
|
/// ActOnCXXMemberDeclarator - This is invoked when a C++ class member
|
|
|
|
|
/// declarator is parsed. 'AS' is the access specifier, 'BW' specifies the
|
|
|
|
|
/// bitfield width if there is one and 'InitExpr' specifies the initializer if
|
2009-04-13 06:37:57 +08:00
|
|
|
|
/// any.
|
2009-03-29 03:18:32 +08:00
|
|
|
|
Sema::DeclPtrTy
|
2008-07-01 18:37:29 +08:00
|
|
|
|
Sema::ActOnCXXMemberDeclarator(Scope *S, AccessSpecifier AS, Declarator &D,
|
2009-04-13 01:16:29 +08:00
|
|
|
|
ExprTy *BW, ExprTy *InitExpr, bool Deleted) {
|
2008-07-01 18:37:29 +08:00
|
|
|
|
const DeclSpec &DS = D.getDeclSpec();
|
2008-11-18 06:58:34 +08:00
|
|
|
|
DeclarationName Name = GetNameForDeclarator(D);
|
2008-07-01 18:37:29 +08:00
|
|
|
|
Expr *BitWidth = static_cast<Expr*>(BW);
|
|
|
|
|
Expr *Init = static_cast<Expr*>(InitExpr);
|
|
|
|
|
SourceLocation Loc = D.getIdentifierLoc();
|
|
|
|
|
|
2008-11-15 07:42:31 +08:00
|
|
|
|
bool isFunc = D.isFunctionDeclarator();
|
|
|
|
|
|
2008-07-01 18:37:29 +08:00
|
|
|
|
// C++ 9.2p6: A member shall not be declared to have automatic storage
|
|
|
|
|
// duration (auto, register) or with the extern storage-class-specifier.
|
2008-11-15 07:42:31 +08:00
|
|
|
|
// C++ 7.1.1p8: The mutable specifier can be applied only to names of class
|
|
|
|
|
// data members and cannot be applied to names declared const or static,
|
|
|
|
|
// and cannot be applied to reference members.
|
2008-07-01 18:37:29 +08:00
|
|
|
|
switch (DS.getStorageClassSpec()) {
|
|
|
|
|
case DeclSpec::SCS_unspecified:
|
|
|
|
|
case DeclSpec::SCS_typedef:
|
|
|
|
|
case DeclSpec::SCS_static:
|
|
|
|
|
// FALL THROUGH.
|
|
|
|
|
break;
|
2008-11-15 07:42:31 +08:00
|
|
|
|
case DeclSpec::SCS_mutable:
|
|
|
|
|
if (isFunc) {
|
|
|
|
|
if (DS.getStorageClassSpecLoc().isValid())
|
2008-11-19 13:08:23 +08:00
|
|
|
|
Diag(DS.getStorageClassSpecLoc(), diag::err_mutable_function);
|
2008-11-15 07:42:31 +08:00
|
|
|
|
else
|
2008-11-19 13:08:23 +08:00
|
|
|
|
Diag(DS.getThreadSpecLoc(), diag::err_mutable_function);
|
|
|
|
|
|
2008-11-18 07:24:37 +08:00
|
|
|
|
// FIXME: It would be nicer if the keyword was ignored only for this
|
|
|
|
|
// declarator. Otherwise we could get follow-up errors.
|
2008-11-15 07:42:31 +08:00
|
|
|
|
D.getMutableDeclSpec().ClearStorageClassSpecs();
|
|
|
|
|
} else {
|
|
|
|
|
QualType T = GetTypeForDeclarator(D, S);
|
|
|
|
|
diag::kind err = static_cast<diag::kind>(0);
|
|
|
|
|
if (T->isReferenceType())
|
|
|
|
|
err = diag::err_mutable_reference;
|
|
|
|
|
else if (T.isConstQualified())
|
|
|
|
|
err = diag::err_mutable_const;
|
|
|
|
|
if (err != 0) {
|
|
|
|
|
if (DS.getStorageClassSpecLoc().isValid())
|
|
|
|
|
Diag(DS.getStorageClassSpecLoc(), err);
|
|
|
|
|
else
|
|
|
|
|
Diag(DS.getThreadSpecLoc(), err);
|
2008-11-18 07:24:37 +08:00
|
|
|
|
// FIXME: It would be nicer if the keyword was ignored only for this
|
|
|
|
|
// declarator. Otherwise we could get follow-up errors.
|
2008-11-15 07:42:31 +08:00
|
|
|
|
D.getMutableDeclSpec().ClearStorageClassSpecs();
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
break;
|
2008-07-01 18:37:29 +08:00
|
|
|
|
default:
|
|
|
|
|
if (DS.getStorageClassSpecLoc().isValid())
|
|
|
|
|
Diag(DS.getStorageClassSpecLoc(),
|
|
|
|
|
diag::err_storageclass_invalid_for_member);
|
|
|
|
|
else
|
|
|
|
|
Diag(DS.getThreadSpecLoc(), diag::err_storageclass_invalid_for_member);
|
|
|
|
|
D.getMutableDeclSpec().ClearStorageClassSpecs();
|
|
|
|
|
}
|
|
|
|
|
|
2008-10-16 04:23:22 +08:00
|
|
|
|
if (!isFunc &&
|
2009-02-09 23:09:02 +08:00
|
|
|
|
D.getDeclSpec().getTypeSpecType() == DeclSpec::TST_typename &&
|
2008-10-16 04:23:22 +08:00
|
|
|
|
D.getNumTypeObjects() == 0) {
|
2008-10-09 06:20:31 +08:00
|
|
|
|
// Check also for this case:
|
|
|
|
|
//
|
|
|
|
|
// typedef int f();
|
|
|
|
|
// f a;
|
|
|
|
|
//
|
2009-02-09 23:09:02 +08:00
|
|
|
|
QualType TDType = QualType::getFromOpaquePtr(DS.getTypeRep());
|
|
|
|
|
isFunc = TDType->isFunctionType();
|
2008-10-09 06:20:31 +08:00
|
|
|
|
}
|
2008-07-01 18:37:29 +08:00
|
|
|
|
|
2008-11-15 07:42:31 +08:00
|
|
|
|
bool isInstField = ((DS.getStorageClassSpec() == DeclSpec::SCS_unspecified ||
|
|
|
|
|
DS.getStorageClassSpec() == DeclSpec::SCS_mutable) &&
|
2008-10-09 06:20:31 +08:00
|
|
|
|
!isFunc);
|
2008-07-01 18:37:29 +08:00
|
|
|
|
|
|
|
|
|
Decl *Member;
|
2009-03-06 06:45:59 +08:00
|
|
|
|
if (isInstField) {
|
2009-03-12 04:50:30 +08:00
|
|
|
|
Member = HandleField(S, cast<CXXRecordDecl>(CurContext), Loc, D, BitWidth,
|
|
|
|
|
AS);
|
2009-03-06 07:03:49 +08:00
|
|
|
|
assert(Member && "HandleField never returns null");
|
2009-03-06 06:45:59 +08:00
|
|
|
|
} else {
|
2009-03-30 00:50:03 +08:00
|
|
|
|
Member = ActOnDeclarator(S, D).getAs<Decl>();
|
2009-03-06 07:03:49 +08:00
|
|
|
|
if (!Member) {
|
|
|
|
|
if (BitWidth) DeleteExpr(BitWidth);
|
2009-03-30 00:50:03 +08:00
|
|
|
|
return DeclPtrTy();
|
2009-03-06 07:03:49 +08:00
|
|
|
|
}
|
2009-03-06 07:01:03 +08:00
|
|
|
|
|
|
|
|
|
// Non-instance-fields can't have a bitfield.
|
|
|
|
|
if (BitWidth) {
|
|
|
|
|
if (Member->isInvalidDecl()) {
|
|
|
|
|
// don't emit another diagnostic.
|
2009-03-12 04:22:50 +08:00
|
|
|
|
} else if (isa<VarDecl>(Member)) {
|
2009-03-06 07:01:03 +08:00
|
|
|
|
// C++ 9.6p3: A bit-field shall not be a static member.
|
|
|
|
|
// "static member 'A' cannot be a bit-field"
|
|
|
|
|
Diag(Loc, diag::err_static_not_bitfield)
|
|
|
|
|
<< Name << BitWidth->getSourceRange();
|
|
|
|
|
} else if (isa<TypedefDecl>(Member)) {
|
|
|
|
|
// "typedef member 'x' cannot be a bit-field"
|
|
|
|
|
Diag(Loc, diag::err_typedef_not_bitfield)
|
|
|
|
|
<< Name << BitWidth->getSourceRange();
|
|
|
|
|
} else {
|
|
|
|
|
// A function typedef ("typedef int f(); f a;").
|
|
|
|
|
// C++ 9.6p3: A bit-field shall have integral or enumeration type.
|
|
|
|
|
Diag(Loc, diag::err_not_integral_type_bitfield)
|
2009-03-12 02:59:21 +08:00
|
|
|
|
<< Name << cast<ValueDecl>(Member)->getType()
|
|
|
|
|
<< BitWidth->getSourceRange();
|
2009-03-06 07:01:03 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
DeleteExpr(BitWidth);
|
|
|
|
|
BitWidth = 0;
|
|
|
|
|
Member->setInvalidDecl();
|
|
|
|
|
}
|
2009-03-12 04:50:30 +08:00
|
|
|
|
|
|
|
|
|
Member->setAccess(AS);
|
2009-03-06 06:45:59 +08:00
|
|
|
|
}
|
2008-07-01 18:37:29 +08:00
|
|
|
|
|
2008-11-18 06:58:34 +08:00
|
|
|
|
assert((Name || isInstField) && "No identifier for non-field ?");
|
2008-07-01 18:37:29 +08:00
|
|
|
|
|
2009-03-12 07:00:04 +08:00
|
|
|
|
if (Init)
|
2009-03-29 03:18:32 +08:00
|
|
|
|
AddInitializerToDecl(DeclPtrTy::make(Member), ExprArg(*this, Init), false);
|
2009-04-13 01:16:29 +08:00
|
|
|
|
if (Deleted) // FIXME: Source location is not very good.
|
|
|
|
|
SetDeclDeleted(DeclPtrTy::make(Member), D.getSourceRange().getBegin());
|
2008-07-01 18:37:29 +08:00
|
|
|
|
|
|
|
|
|
if (isInstField) {
|
2008-12-12 00:49:14 +08:00
|
|
|
|
FieldCollector->Add(cast<FieldDecl>(Member));
|
2009-03-30 00:50:03 +08:00
|
|
|
|
return DeclPtrTy();
|
2008-07-01 18:37:29 +08:00
|
|
|
|
}
|
2009-03-29 03:18:32 +08:00
|
|
|
|
return DeclPtrTy::make(Member);
|
2008-07-01 18:37:29 +08:00
|
|
|
|
}
|
|
|
|
|
|
2008-11-05 12:29:56 +08:00
|
|
|
|
/// ActOnMemInitializer - Handle a C++ member initializer.
|
|
|
|
|
Sema::MemInitResult
|
2009-03-29 03:18:32 +08:00
|
|
|
|
Sema::ActOnMemInitializer(DeclPtrTy ConstructorD,
|
2008-11-05 12:29:56 +08:00
|
|
|
|
Scope *S,
|
|
|
|
|
IdentifierInfo *MemberOrBase,
|
|
|
|
|
SourceLocation IdLoc,
|
|
|
|
|
SourceLocation LParenLoc,
|
|
|
|
|
ExprTy **Args, unsigned NumArgs,
|
|
|
|
|
SourceLocation *CommaLocs,
|
|
|
|
|
SourceLocation RParenLoc) {
|
|
|
|
|
CXXConstructorDecl *Constructor
|
2009-03-29 03:18:32 +08:00
|
|
|
|
= dyn_cast<CXXConstructorDecl>(ConstructorD.getAs<Decl>());
|
2008-11-05 12:29:56 +08:00
|
|
|
|
if (!Constructor) {
|
|
|
|
|
// The user wrote a constructor initializer on a function that is
|
|
|
|
|
// not a C++ constructor. Ignore the error for now, because we may
|
|
|
|
|
// have more member initializers coming; we'll diagnose it just
|
|
|
|
|
// once in ActOnMemInitializers.
|
|
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
CXXRecordDecl *ClassDecl = Constructor->getParent();
|
|
|
|
|
|
|
|
|
|
// C++ [class.base.init]p2:
|
|
|
|
|
// Names in a mem-initializer-id are looked up in the scope of the
|
|
|
|
|
// constructor’s class and, if not found in that scope, are looked
|
|
|
|
|
// up in the scope containing the constructor’s
|
|
|
|
|
// definition. [Note: if the constructor’s class contains a member
|
|
|
|
|
// with the same name as a direct or virtual base class of the
|
|
|
|
|
// class, a mem-initializer-id naming the member or base class and
|
|
|
|
|
// composed of a single identifier refers to the class member. A
|
|
|
|
|
// mem-initializer-id for the hidden base class may be specified
|
|
|
|
|
// using a qualified name. ]
|
|
|
|
|
// Look for a member, first.
|
2008-12-12 00:49:14 +08:00
|
|
|
|
FieldDecl *Member = 0;
|
2009-04-10 05:40:53 +08:00
|
|
|
|
DeclContext::lookup_result Result
|
|
|
|
|
= ClassDecl->lookup(Context, MemberOrBase);
|
2008-12-12 00:49:14 +08:00
|
|
|
|
if (Result.first != Result.second)
|
|
|
|
|
Member = dyn_cast<FieldDecl>(*Result.first);
|
2008-11-05 12:29:56 +08:00
|
|
|
|
|
|
|
|
|
// FIXME: Handle members of an anonymous union.
|
|
|
|
|
|
|
|
|
|
if (Member) {
|
|
|
|
|
// FIXME: Perform direct initialization of the member.
|
|
|
|
|
return new CXXBaseOrMemberInitializer(Member, (Expr **)Args, NumArgs);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// It didn't name a member, so see if it names a class.
|
2009-02-05 01:00:24 +08:00
|
|
|
|
TypeTy *BaseTy = getTypeName(*MemberOrBase, IdLoc, S, 0/*SS*/);
|
2008-11-05 12:29:56 +08:00
|
|
|
|
if (!BaseTy)
|
2008-11-19 16:23:25 +08:00
|
|
|
|
return Diag(IdLoc, diag::err_mem_init_not_member_or_class)
|
|
|
|
|
<< MemberOrBase << SourceRange(IdLoc, RParenLoc);
|
2008-11-05 12:29:56 +08:00
|
|
|
|
|
2009-02-09 23:09:02 +08:00
|
|
|
|
QualType BaseType = QualType::getFromOpaquePtr(BaseTy);
|
2008-11-05 12:29:56 +08:00
|
|
|
|
if (!BaseType->isRecordType())
|
2008-11-19 16:23:25 +08:00
|
|
|
|
return Diag(IdLoc, diag::err_base_init_does_not_name_class)
|
2008-11-24 05:45:46 +08:00
|
|
|
|
<< BaseType << SourceRange(IdLoc, RParenLoc);
|
2008-11-05 12:29:56 +08:00
|
|
|
|
|
|
|
|
|
// C++ [class.base.init]p2:
|
|
|
|
|
// [...] Unless the mem-initializer-id names a nonstatic data
|
|
|
|
|
// member of the constructor’s class or a direct or virtual base
|
|
|
|
|
// of that class, the mem-initializer is ill-formed. A
|
|
|
|
|
// mem-initializer-list can initialize a base class using any
|
|
|
|
|
// name that denotes that base class type.
|
|
|
|
|
|
|
|
|
|
// First, check for a direct base class.
|
|
|
|
|
const CXXBaseSpecifier *DirectBaseSpec = 0;
|
|
|
|
|
for (CXXRecordDecl::base_class_const_iterator Base = ClassDecl->bases_begin();
|
|
|
|
|
Base != ClassDecl->bases_end(); ++Base) {
|
|
|
|
|
if (Context.getCanonicalType(BaseType).getUnqualifiedType() ==
|
|
|
|
|
Context.getCanonicalType(Base->getType()).getUnqualifiedType()) {
|
|
|
|
|
// We found a direct base of this type. That's what we're
|
|
|
|
|
// initializing.
|
|
|
|
|
DirectBaseSpec = &*Base;
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Check for a virtual base class.
|
2009-05-16 15:39:55 +08:00
|
|
|
|
// FIXME: We might be able to short-circuit this if we know in advance that
|
|
|
|
|
// there are no virtual bases.
|
2008-11-05 12:29:56 +08:00
|
|
|
|
const CXXBaseSpecifier *VirtualBaseSpec = 0;
|
|
|
|
|
if (!DirectBaseSpec || !DirectBaseSpec->isVirtual()) {
|
|
|
|
|
// We haven't found a base yet; search the class hierarchy for a
|
|
|
|
|
// virtual base class.
|
|
|
|
|
BasePaths Paths(/*FindAmbiguities=*/true, /*RecordPaths=*/true,
|
|
|
|
|
/*DetectVirtual=*/false);
|
|
|
|
|
if (IsDerivedFrom(Context.getTypeDeclType(ClassDecl), BaseType, Paths)) {
|
|
|
|
|
for (BasePaths::paths_iterator Path = Paths.begin();
|
|
|
|
|
Path != Paths.end(); ++Path) {
|
|
|
|
|
if (Path->back().Base->isVirtual()) {
|
|
|
|
|
VirtualBaseSpec = Path->back().Base;
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// C++ [base.class.init]p2:
|
|
|
|
|
// If a mem-initializer-id is ambiguous because it designates both
|
|
|
|
|
// a direct non-virtual base class and an inherited virtual base
|
|
|
|
|
// class, the mem-initializer is ill-formed.
|
|
|
|
|
if (DirectBaseSpec && VirtualBaseSpec)
|
2008-11-19 16:23:25 +08:00
|
|
|
|
return Diag(IdLoc, diag::err_base_init_direct_and_virtual)
|
|
|
|
|
<< MemberOrBase << SourceRange(IdLoc, RParenLoc);
|
2008-11-05 12:29:56 +08:00
|
|
|
|
|
|
|
|
|
return new CXXBaseOrMemberInitializer(BaseType, (Expr **)Args, NumArgs);
|
|
|
|
|
}
|
|
|
|
|
|
2009-03-29 03:18:32 +08:00
|
|
|
|
void Sema::ActOnMemInitializers(DeclPtrTy ConstructorDecl,
|
2009-03-25 10:58:17 +08:00
|
|
|
|
SourceLocation ColonLoc,
|
|
|
|
|
MemInitTy **MemInits, unsigned NumMemInits) {
|
|
|
|
|
CXXConstructorDecl *Constructor =
|
2009-03-29 03:18:32 +08:00
|
|
|
|
dyn_cast<CXXConstructorDecl>(ConstructorDecl.getAs<Decl>());
|
2009-03-25 10:58:17 +08:00
|
|
|
|
|
|
|
|
|
if (!Constructor) {
|
|
|
|
|
Diag(ColonLoc, diag::err_only_constructors_take_base_inits);
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2009-03-22 09:52:17 +08:00
|
|
|
|
namespace {
|
|
|
|
|
/// PureVirtualMethodCollector - traverses a class and its superclasses
|
|
|
|
|
/// and determines if it has any pure virtual methods.
|
|
|
|
|
class VISIBILITY_HIDDEN PureVirtualMethodCollector {
|
|
|
|
|
ASTContext &Context;
|
|
|
|
|
|
2009-03-23 05:28:55 +08:00
|
|
|
|
public:
|
2009-03-22 09:52:17 +08:00
|
|
|
|
typedef llvm::SmallVector<const CXXMethodDecl*, 8> MethodList;
|
2009-03-23 05:28:55 +08:00
|
|
|
|
|
|
|
|
|
private:
|
2009-03-22 09:52:17 +08:00
|
|
|
|
MethodList Methods;
|
|
|
|
|
|
|
|
|
|
void Collect(const CXXRecordDecl* RD, MethodList& Methods);
|
|
|
|
|
|
|
|
|
|
public:
|
|
|
|
|
PureVirtualMethodCollector(ASTContext &Ctx, const CXXRecordDecl* RD)
|
|
|
|
|
: Context(Ctx) {
|
|
|
|
|
|
|
|
|
|
MethodList List;
|
|
|
|
|
Collect(RD, List);
|
|
|
|
|
|
|
|
|
|
// Copy the temporary list to methods, and make sure to ignore any
|
|
|
|
|
// null entries.
|
|
|
|
|
for (size_t i = 0, e = List.size(); i != e; ++i) {
|
|
|
|
|
if (List[i])
|
|
|
|
|
Methods.push_back(List[i]);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2009-03-23 04:18:17 +08:00
|
|
|
|
bool empty() const { return Methods.empty(); }
|
|
|
|
|
|
|
|
|
|
MethodList::const_iterator methods_begin() { return Methods.begin(); }
|
|
|
|
|
MethodList::const_iterator methods_end() { return Methods.end(); }
|
2009-03-22 09:52:17 +08:00
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
void PureVirtualMethodCollector::Collect(const CXXRecordDecl* RD,
|
|
|
|
|
MethodList& Methods) {
|
|
|
|
|
// First, collect the pure virtual methods for the base classes.
|
|
|
|
|
for (CXXRecordDecl::base_class_const_iterator Base = RD->bases_begin(),
|
|
|
|
|
BaseEnd = RD->bases_end(); Base != BaseEnd; ++Base) {
|
|
|
|
|
if (const RecordType *RT = Base->getType()->getAsRecordType()) {
|
2009-03-29 13:01:10 +08:00
|
|
|
|
const CXXRecordDecl *BaseDecl = cast<CXXRecordDecl>(RT->getDecl());
|
2009-03-22 09:52:17 +08:00
|
|
|
|
if (BaseDecl && BaseDecl->isAbstract())
|
|
|
|
|
Collect(BaseDecl, Methods);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Next, zero out any pure virtual methods that this class overrides.
|
2009-05-17 08:00:05 +08:00
|
|
|
|
typedef llvm::SmallPtrSet<const CXXMethodDecl*, 4> MethodSetTy;
|
|
|
|
|
|
|
|
|
|
MethodSetTy OverriddenMethods;
|
|
|
|
|
size_t MethodsSize = Methods.size();
|
|
|
|
|
|
2009-04-10 05:40:53 +08:00
|
|
|
|
for (RecordDecl::decl_iterator i = RD->decls_begin(Context),
|
2009-05-17 08:00:05 +08:00
|
|
|
|
e = RD->decls_end(Context);
|
2009-03-22 09:52:17 +08:00
|
|
|
|
i != e; ++i) {
|
2009-05-17 08:00:05 +08:00
|
|
|
|
// Traverse the record, looking for methods.
|
2009-03-22 09:52:17 +08:00
|
|
|
|
if (CXXMethodDecl *MD = dyn_cast<CXXMethodDecl>(*i)) {
|
2009-05-17 08:00:05 +08:00
|
|
|
|
// If the method is pre virtual, add it to the methods vector.
|
|
|
|
|
if (MD->isPure()) {
|
2009-03-22 09:52:17 +08:00
|
|
|
|
Methods.push_back(MD);
|
2009-05-17 08:00:05 +08:00
|
|
|
|
continue;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Otherwise, record all the overridden methods in our set.
|
|
|
|
|
for (CXXMethodDecl::method_iterator I = MD->begin_overridden_methods(),
|
|
|
|
|
E = MD->end_overridden_methods(); I != E; ++I) {
|
|
|
|
|
// Keep track of the overridden methods.
|
|
|
|
|
OverriddenMethods.insert(*I);
|
|
|
|
|
}
|
2009-03-22 09:52:17 +08:00
|
|
|
|
}
|
|
|
|
|
}
|
2009-05-17 08:00:05 +08:00
|
|
|
|
|
|
|
|
|
// Now go through the methods and zero out all the ones we know are
|
|
|
|
|
// overridden.
|
|
|
|
|
for (size_t i = 0, e = MethodsSize; i != e; ++i) {
|
|
|
|
|
if (OverriddenMethods.count(Methods[i]))
|
|
|
|
|
Methods[i] = 0;
|
|
|
|
|
}
|
|
|
|
|
|
2009-03-22 09:52:17 +08:00
|
|
|
|
}
|
|
|
|
|
}
|
2008-11-05 12:29:56 +08:00
|
|
|
|
|
2009-03-23 04:18:17 +08:00
|
|
|
|
bool Sema::RequireNonAbstractType(SourceLocation Loc, QualType T,
|
2009-03-25 01:23:42 +08:00
|
|
|
|
unsigned DiagID, AbstractDiagSelID SelID,
|
|
|
|
|
const CXXRecordDecl *CurrentRD) {
|
2009-03-23 04:18:17 +08:00
|
|
|
|
|
|
|
|
|
if (!getLangOptions().CPlusPlus)
|
|
|
|
|
return false;
|
2009-03-24 03:10:31 +08:00
|
|
|
|
|
|
|
|
|
if (const ArrayType *AT = Context.getAsArrayType(T))
|
2009-03-25 01:23:42 +08:00
|
|
|
|
return RequireNonAbstractType(Loc, AT->getElementType(), DiagID, SelID,
|
|
|
|
|
CurrentRD);
|
2009-03-24 09:46:45 +08:00
|
|
|
|
|
|
|
|
|
if (const PointerType *PT = T->getAsPointerType()) {
|
|
|
|
|
// Find the innermost pointer type.
|
|
|
|
|
while (const PointerType *T = PT->getPointeeType()->getAsPointerType())
|
|
|
|
|
PT = T;
|
2009-03-23 04:18:17 +08:00
|
|
|
|
|
2009-03-24 09:46:45 +08:00
|
|
|
|
if (const ArrayType *AT = Context.getAsArrayType(PT->getPointeeType()))
|
2009-03-25 01:23:42 +08:00
|
|
|
|
return RequireNonAbstractType(Loc, AT->getElementType(), DiagID, SelID,
|
|
|
|
|
CurrentRD);
|
2009-03-24 09:46:45 +08:00
|
|
|
|
}
|
|
|
|
|
|
2009-03-23 04:18:17 +08:00
|
|
|
|
const RecordType *RT = T->getAsRecordType();
|
|
|
|
|
if (!RT)
|
|
|
|
|
return false;
|
|
|
|
|
|
|
|
|
|
const CXXRecordDecl *RD = dyn_cast<CXXRecordDecl>(RT->getDecl());
|
|
|
|
|
if (!RD)
|
|
|
|
|
return false;
|
|
|
|
|
|
2009-03-25 01:23:42 +08:00
|
|
|
|
if (CurrentRD && CurrentRD != RD)
|
|
|
|
|
return false;
|
|
|
|
|
|
2009-03-23 04:18:17 +08:00
|
|
|
|
if (!RD->isAbstract())
|
|
|
|
|
return false;
|
|
|
|
|
|
2009-03-24 01:49:10 +08:00
|
|
|
|
Diag(Loc, DiagID) << RD->getDeclName() << SelID;
|
2009-03-23 04:18:17 +08:00
|
|
|
|
|
|
|
|
|
// Check if we've already emitted the list of pure virtual functions for this
|
|
|
|
|
// class.
|
|
|
|
|
if (PureVirtualClassDiagSet && PureVirtualClassDiagSet->count(RD))
|
|
|
|
|
return true;
|
|
|
|
|
|
|
|
|
|
PureVirtualMethodCollector Collector(Context, RD);
|
|
|
|
|
|
|
|
|
|
for (PureVirtualMethodCollector::MethodList::const_iterator I =
|
|
|
|
|
Collector.methods_begin(), E = Collector.methods_end(); I != E; ++I) {
|
|
|
|
|
const CXXMethodDecl *MD = *I;
|
|
|
|
|
|
|
|
|
|
Diag(MD->getLocation(), diag::note_pure_virtual_function) <<
|
|
|
|
|
MD->getDeclName();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (!PureVirtualClassDiagSet)
|
|
|
|
|
PureVirtualClassDiagSet.reset(new RecordDeclSetTy);
|
|
|
|
|
PureVirtualClassDiagSet->insert(RD);
|
|
|
|
|
|
|
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
|
2009-03-24 09:19:16 +08:00
|
|
|
|
namespace {
|
|
|
|
|
class VISIBILITY_HIDDEN AbstractClassUsageDiagnoser
|
|
|
|
|
: public DeclVisitor<AbstractClassUsageDiagnoser, bool> {
|
|
|
|
|
Sema &SemaRef;
|
|
|
|
|
CXXRecordDecl *AbstractClass;
|
|
|
|
|
|
2009-03-25 01:23:42 +08:00
|
|
|
|
bool VisitDeclContext(const DeclContext *DC) {
|
2009-03-24 09:19:16 +08:00
|
|
|
|
bool Invalid = false;
|
|
|
|
|
|
2009-04-10 05:40:53 +08:00
|
|
|
|
for (CXXRecordDecl::decl_iterator I = DC->decls_begin(SemaRef.Context),
|
|
|
|
|
E = DC->decls_end(SemaRef.Context); I != E; ++I)
|
2009-03-24 09:19:16 +08:00
|
|
|
|
Invalid |= Visit(*I);
|
2009-03-25 01:23:42 +08:00
|
|
|
|
|
2009-03-24 09:19:16 +08:00
|
|
|
|
return Invalid;
|
|
|
|
|
}
|
2009-03-25 01:23:42 +08:00
|
|
|
|
|
|
|
|
|
public:
|
|
|
|
|
AbstractClassUsageDiagnoser(Sema& SemaRef, CXXRecordDecl *ac)
|
|
|
|
|
: SemaRef(SemaRef), AbstractClass(ac) {
|
|
|
|
|
Visit(SemaRef.Context.getTranslationUnitDecl());
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
bool VisitFunctionDecl(const FunctionDecl *FD) {
|
|
|
|
|
if (FD->isThisDeclarationADefinition()) {
|
|
|
|
|
// No need to do the check if we're in a definition, because it requires
|
|
|
|
|
// that the return/param types are complete.
|
|
|
|
|
// because that requires
|
|
|
|
|
return VisitDeclContext(FD);
|
|
|
|
|
}
|
|
|
|
|
|
2009-03-24 09:19:16 +08:00
|
|
|
|
// Check the return type.
|
2009-03-25 01:23:42 +08:00
|
|
|
|
QualType RTy = FD->getType()->getAsFunctionType()->getResultType();
|
2009-03-24 09:19:16 +08:00
|
|
|
|
bool Invalid =
|
2009-03-25 01:23:42 +08:00
|
|
|
|
SemaRef.RequireNonAbstractType(FD->getLocation(), RTy,
|
2009-03-24 09:19:16 +08:00
|
|
|
|
diag::err_abstract_type_in_decl,
|
2009-03-25 01:23:42 +08:00
|
|
|
|
Sema::AbstractReturnType,
|
|
|
|
|
AbstractClass);
|
2009-03-24 09:19:16 +08:00
|
|
|
|
|
2009-03-25 01:23:42 +08:00
|
|
|
|
for (FunctionDecl::param_const_iterator I = FD->param_begin(),
|
|
|
|
|
E = FD->param_end(); I != E; ++I) {
|
2009-03-24 09:19:16 +08:00
|
|
|
|
const ParmVarDecl *VD = *I;
|
|
|
|
|
Invalid |=
|
|
|
|
|
SemaRef.RequireNonAbstractType(VD->getLocation(),
|
|
|
|
|
VD->getOriginalType(),
|
|
|
|
|
diag::err_abstract_type_in_decl,
|
2009-03-25 01:23:42 +08:00
|
|
|
|
Sema::AbstractParamType,
|
|
|
|
|
AbstractClass);
|
2009-03-24 09:19:16 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return Invalid;
|
|
|
|
|
}
|
2009-03-25 01:23:42 +08:00
|
|
|
|
|
|
|
|
|
bool VisitDecl(const Decl* D) {
|
|
|
|
|
if (const DeclContext *DC = dyn_cast<DeclContext>(D))
|
|
|
|
|
return VisitDeclContext(DC);
|
|
|
|
|
|
|
|
|
|
return false;
|
|
|
|
|
}
|
2009-03-24 09:19:16 +08:00
|
|
|
|
};
|
|
|
|
|
}
|
|
|
|
|
|
2008-07-01 18:37:29 +08:00
|
|
|
|
void Sema::ActOnFinishCXXMemberSpecification(Scope* S, SourceLocation RLoc,
|
2009-03-29 03:18:32 +08:00
|
|
|
|
DeclPtrTy TagDecl,
|
2008-07-01 18:37:29 +08:00
|
|
|
|
SourceLocation LBrac,
|
|
|
|
|
SourceLocation RBrac) {
|
2009-05-12 03:58:34 +08:00
|
|
|
|
AdjustDeclIfTemplate(TagDecl);
|
2008-07-01 18:37:29 +08:00
|
|
|
|
ActOnFields(S, RLoc, TagDecl,
|
2009-03-29 03:18:32 +08:00
|
|
|
|
(DeclPtrTy*)FieldCollector->getCurFields(),
|
2008-10-03 10:03:53 +08:00
|
|
|
|
FieldCollector->getCurNumFields(), LBrac, RBrac, 0);
|
2009-03-03 12:44:36 +08:00
|
|
|
|
|
2009-03-29 03:18:32 +08:00
|
|
|
|
CXXRecordDecl *RD = cast<CXXRecordDecl>(TagDecl.getAs<Decl>());
|
2009-03-22 09:52:17 +08:00
|
|
|
|
if (!RD->isAbstract()) {
|
|
|
|
|
// Collect all the pure virtual methods and see if this is an abstract
|
|
|
|
|
// class after all.
|
|
|
|
|
PureVirtualMethodCollector Collector(Context, RD);
|
|
|
|
|
if (!Collector.empty())
|
|
|
|
|
RD->setAbstract(true);
|
|
|
|
|
}
|
|
|
|
|
|
2009-03-25 01:23:42 +08:00
|
|
|
|
if (RD->isAbstract())
|
|
|
|
|
AbstractClassUsageDiagnoser(*this, RD);
|
2009-03-24 09:19:16 +08:00
|
|
|
|
|
2009-04-17 10:34:54 +08:00
|
|
|
|
if (RD->hasTrivialConstructor() || RD->hasTrivialDestructor()) {
|
2009-04-16 08:08:20 +08:00
|
|
|
|
for (RecordDecl::field_iterator i = RD->field_begin(Context),
|
|
|
|
|
e = RD->field_end(Context); i != e; ++i) {
|
|
|
|
|
// All the nonstatic data members must have trivial constructors.
|
|
|
|
|
QualType FTy = i->getType();
|
|
|
|
|
while (const ArrayType *AT = Context.getAsArrayType(FTy))
|
|
|
|
|
FTy = AT->getElementType();
|
|
|
|
|
|
|
|
|
|
if (const RecordType *RT = FTy->getAsRecordType()) {
|
|
|
|
|
CXXRecordDecl *FieldRD = cast<CXXRecordDecl>(RT->getDecl());
|
2009-04-17 10:34:54 +08:00
|
|
|
|
|
|
|
|
|
if (!FieldRD->hasTrivialConstructor())
|
2009-04-16 08:08:20 +08:00
|
|
|
|
RD->setHasTrivialConstructor(false);
|
2009-04-17 10:34:54 +08:00
|
|
|
|
if (!FieldRD->hasTrivialDestructor())
|
|
|
|
|
RD->setHasTrivialDestructor(false);
|
|
|
|
|
|
|
|
|
|
// If RD has neither a trivial constructor nor a trivial destructor
|
|
|
|
|
// we don't need to continue checking.
|
|
|
|
|
if (!RD->hasTrivialConstructor() && !RD->hasTrivialDestructor())
|
2009-04-16 08:08:20 +08:00
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2009-05-12 03:58:34 +08:00
|
|
|
|
if (!RD->isDependentType())
|
2009-03-22 09:52:17 +08:00
|
|
|
|
AddImplicitlyDeclaredMembersToClass(RD);
|
2008-07-01 18:37:29 +08:00
|
|
|
|
}
|
|
|
|
|
|
2008-11-04 01:51:48 +08:00
|
|
|
|
/// AddImplicitlyDeclaredMembersToClass - Adds any implicitly-declared
|
|
|
|
|
/// special functions, such as the default constructor, copy
|
|
|
|
|
/// constructor, or destructor, to the given C++ class (C++
|
|
|
|
|
/// [special]p1). This routine can only be executed just before the
|
|
|
|
|
/// definition of the class is complete.
|
|
|
|
|
void Sema::AddImplicitlyDeclaredMembersToClass(CXXRecordDecl *ClassDecl) {
|
2008-11-17 22:58:09 +08:00
|
|
|
|
QualType ClassType = Context.getTypeDeclType(ClassDecl);
|
|
|
|
|
ClassType = Context.getCanonicalType(ClassType);
|
|
|
|
|
|
2009-05-28 06:11:52 +08:00
|
|
|
|
// FIXME: Implicit declarations have exception specifications, which are
|
|
|
|
|
// the union of the specifications of the implicitly called functions.
|
|
|
|
|
|
2008-11-04 01:51:48 +08:00
|
|
|
|
if (!ClassDecl->hasUserDeclaredConstructor()) {
|
|
|
|
|
// C++ [class.ctor]p5:
|
|
|
|
|
// A default constructor for a class X is a constructor of class X
|
|
|
|
|
// that can be called without an argument. If there is no
|
|
|
|
|
// user-declared constructor for class X, a default constructor is
|
|
|
|
|
// implicitly declared. An implicitly-declared default constructor
|
|
|
|
|
// is an inline public member of its class.
|
2008-11-17 22:58:09 +08:00
|
|
|
|
DeclarationName Name
|
|
|
|
|
= Context.DeclarationNames.getCXXConstructorName(ClassType);
|
2008-11-04 01:51:48 +08:00
|
|
|
|
CXXConstructorDecl *DefaultCon =
|
|
|
|
|
CXXConstructorDecl::Create(Context, ClassDecl,
|
2008-11-17 22:58:09 +08:00
|
|
|
|
ClassDecl->getLocation(), Name,
|
2008-11-04 01:51:48 +08:00
|
|
|
|
Context.getFunctionType(Context.VoidTy,
|
|
|
|
|
0, 0, false, 0),
|
|
|
|
|
/*isExplicit=*/false,
|
|
|
|
|
/*isInline=*/true,
|
|
|
|
|
/*isImplicitlyDeclared=*/true);
|
|
|
|
|
DefaultCon->setAccess(AS_public);
|
2009-01-08 03:46:03 +08:00
|
|
|
|
DefaultCon->setImplicit();
|
2009-04-10 05:40:53 +08:00
|
|
|
|
ClassDecl->addDecl(Context, DefaultCon);
|
2008-12-16 05:24:18 +08:00
|
|
|
|
|
|
|
|
|
// Notify the class that we've added a constructor.
|
|
|
|
|
ClassDecl->addedConstructor(Context, DefaultCon);
|
2008-11-04 01:51:48 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (!ClassDecl->hasUserDeclaredCopyConstructor()) {
|
|
|
|
|
// C++ [class.copy]p4:
|
|
|
|
|
// If the class definition does not explicitly declare a copy
|
|
|
|
|
// constructor, one is declared implicitly.
|
|
|
|
|
|
|
|
|
|
// C++ [class.copy]p5:
|
|
|
|
|
// The implicitly-declared copy constructor for a class X will
|
|
|
|
|
// have the form
|
|
|
|
|
//
|
|
|
|
|
// X::X(const X&)
|
|
|
|
|
//
|
|
|
|
|
// if
|
|
|
|
|
bool HasConstCopyConstructor = true;
|
|
|
|
|
|
|
|
|
|
// -- each direct or virtual base class B of X has a copy
|
|
|
|
|
// constructor whose first parameter is of type const B& or
|
|
|
|
|
// const volatile B&, and
|
|
|
|
|
for (CXXRecordDecl::base_class_iterator Base = ClassDecl->bases_begin();
|
|
|
|
|
HasConstCopyConstructor && Base != ClassDecl->bases_end(); ++Base) {
|
|
|
|
|
const CXXRecordDecl *BaseClassDecl
|
|
|
|
|
= cast<CXXRecordDecl>(Base->getType()->getAsRecordType()->getDecl());
|
|
|
|
|
HasConstCopyConstructor
|
|
|
|
|
= BaseClassDecl->hasConstCopyConstructor(Context);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// -- for all the nonstatic data members of X that are of a
|
|
|
|
|
// class type M (or array thereof), each such class type
|
|
|
|
|
// has a copy constructor whose first parameter is of type
|
|
|
|
|
// const M& or const volatile M&.
|
2009-04-10 05:40:53 +08:00
|
|
|
|
for (CXXRecordDecl::field_iterator Field = ClassDecl->field_begin(Context);
|
|
|
|
|
HasConstCopyConstructor && Field != ClassDecl->field_end(Context);
|
|
|
|
|
++Field) {
|
2008-11-04 01:51:48 +08:00
|
|
|
|
QualType FieldType = (*Field)->getType();
|
|
|
|
|
if (const ArrayType *Array = Context.getAsArrayType(FieldType))
|
|
|
|
|
FieldType = Array->getElementType();
|
|
|
|
|
if (const RecordType *FieldClassType = FieldType->getAsRecordType()) {
|
|
|
|
|
const CXXRecordDecl *FieldClassDecl
|
|
|
|
|
= cast<CXXRecordDecl>(FieldClassType->getDecl());
|
|
|
|
|
HasConstCopyConstructor
|
|
|
|
|
= FieldClassDecl->hasConstCopyConstructor(Context);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2009-01-06 04:52:13 +08:00
|
|
|
|
// Otherwise, the implicitly declared copy constructor will have
|
|
|
|
|
// the form
|
2008-11-04 01:51:48 +08:00
|
|
|
|
//
|
|
|
|
|
// X::X(X&)
|
2009-01-06 04:52:13 +08:00
|
|
|
|
QualType ArgType = ClassType;
|
2008-11-04 01:51:48 +08:00
|
|
|
|
if (HasConstCopyConstructor)
|
|
|
|
|
ArgType = ArgType.withConst();
|
2009-03-17 07:22:08 +08:00
|
|
|
|
ArgType = Context.getLValueReferenceType(ArgType);
|
2008-11-04 01:51:48 +08:00
|
|
|
|
|
2009-01-06 04:52:13 +08:00
|
|
|
|
// An implicitly-declared copy constructor is an inline public
|
|
|
|
|
// member of its class.
|
2008-11-17 22:58:09 +08:00
|
|
|
|
DeclarationName Name
|
|
|
|
|
= Context.DeclarationNames.getCXXConstructorName(ClassType);
|
2008-11-04 01:51:48 +08:00
|
|
|
|
CXXConstructorDecl *CopyConstructor
|
|
|
|
|
= CXXConstructorDecl::Create(Context, ClassDecl,
|
2008-11-17 22:58:09 +08:00
|
|
|
|
ClassDecl->getLocation(), Name,
|
2008-11-04 01:51:48 +08:00
|
|
|
|
Context.getFunctionType(Context.VoidTy,
|
|
|
|
|
&ArgType, 1,
|
|
|
|
|
false, 0),
|
|
|
|
|
/*isExplicit=*/false,
|
|
|
|
|
/*isInline=*/true,
|
|
|
|
|
/*isImplicitlyDeclared=*/true);
|
|
|
|
|
CopyConstructor->setAccess(AS_public);
|
2009-01-08 03:46:03 +08:00
|
|
|
|
CopyConstructor->setImplicit();
|
2008-11-04 01:51:48 +08:00
|
|
|
|
|
|
|
|
|
// Add the parameter to the constructor.
|
|
|
|
|
ParmVarDecl *FromParam = ParmVarDecl::Create(Context, CopyConstructor,
|
|
|
|
|
ClassDecl->getLocation(),
|
|
|
|
|
/*IdentifierInfo=*/0,
|
2009-01-20 09:17:11 +08:00
|
|
|
|
ArgType, VarDecl::None, 0);
|
2009-01-14 08:42:25 +08:00
|
|
|
|
CopyConstructor->setParams(Context, &FromParam, 1);
|
2008-11-04 01:51:48 +08:00
|
|
|
|
|
2008-12-16 05:24:18 +08:00
|
|
|
|
ClassDecl->addedConstructor(Context, CopyConstructor);
|
2009-04-10 05:40:53 +08:00
|
|
|
|
ClassDecl->addDecl(Context, CopyConstructor);
|
2008-11-04 01:51:48 +08:00
|
|
|
|
}
|
|
|
|
|
|
2009-01-06 04:52:13 +08:00
|
|
|
|
if (!ClassDecl->hasUserDeclaredCopyAssignment()) {
|
|
|
|
|
// Note: The following rules are largely analoguous to the copy
|
|
|
|
|
// constructor rules. Note that virtual bases are not taken into account
|
|
|
|
|
// for determining the argument type of the operator. Note also that
|
|
|
|
|
// operators taking an object instead of a reference are allowed.
|
|
|
|
|
//
|
|
|
|
|
// C++ [class.copy]p10:
|
|
|
|
|
// If the class definition does not explicitly declare a copy
|
|
|
|
|
// assignment operator, one is declared implicitly.
|
|
|
|
|
// The implicitly-defined copy assignment operator for a class X
|
|
|
|
|
// will have the form
|
|
|
|
|
//
|
|
|
|
|
// X& X::operator=(const X&)
|
|
|
|
|
//
|
|
|
|
|
// if
|
|
|
|
|
bool HasConstCopyAssignment = true;
|
|
|
|
|
|
|
|
|
|
// -- each direct base class B of X has a copy assignment operator
|
|
|
|
|
// whose parameter is of type const B&, const volatile B& or B,
|
|
|
|
|
// and
|
|
|
|
|
for (CXXRecordDecl::base_class_iterator Base = ClassDecl->bases_begin();
|
|
|
|
|
HasConstCopyAssignment && Base != ClassDecl->bases_end(); ++Base) {
|
|
|
|
|
const CXXRecordDecl *BaseClassDecl
|
|
|
|
|
= cast<CXXRecordDecl>(Base->getType()->getAsRecordType()->getDecl());
|
|
|
|
|
HasConstCopyAssignment = BaseClassDecl->hasConstCopyAssignment(Context);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// -- for all the nonstatic data members of X that are of a class
|
|
|
|
|
// type M (or array thereof), each such class type has a copy
|
|
|
|
|
// assignment operator whose parameter is of type const M&,
|
|
|
|
|
// const volatile M& or M.
|
2009-04-10 05:40:53 +08:00
|
|
|
|
for (CXXRecordDecl::field_iterator Field = ClassDecl->field_begin(Context);
|
|
|
|
|
HasConstCopyAssignment && Field != ClassDecl->field_end(Context);
|
|
|
|
|
++Field) {
|
2009-01-06 04:52:13 +08:00
|
|
|
|
QualType FieldType = (*Field)->getType();
|
|
|
|
|
if (const ArrayType *Array = Context.getAsArrayType(FieldType))
|
|
|
|
|
FieldType = Array->getElementType();
|
|
|
|
|
if (const RecordType *FieldClassType = FieldType->getAsRecordType()) {
|
|
|
|
|
const CXXRecordDecl *FieldClassDecl
|
|
|
|
|
= cast<CXXRecordDecl>(FieldClassType->getDecl());
|
|
|
|
|
HasConstCopyAssignment
|
|
|
|
|
= FieldClassDecl->hasConstCopyAssignment(Context);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Otherwise, the implicitly declared copy assignment operator will
|
|
|
|
|
// have the form
|
|
|
|
|
//
|
|
|
|
|
// X& X::operator=(X&)
|
|
|
|
|
QualType ArgType = ClassType;
|
2009-03-17 07:22:08 +08:00
|
|
|
|
QualType RetType = Context.getLValueReferenceType(ArgType);
|
2009-01-06 04:52:13 +08:00
|
|
|
|
if (HasConstCopyAssignment)
|
|
|
|
|
ArgType = ArgType.withConst();
|
2009-03-17 07:22:08 +08:00
|
|
|
|
ArgType = Context.getLValueReferenceType(ArgType);
|
2009-01-06 04:52:13 +08:00
|
|
|
|
|
|
|
|
|
// An implicitly-declared copy assignment operator is an inline public
|
|
|
|
|
// member of its class.
|
|
|
|
|
DeclarationName Name =
|
|
|
|
|
Context.DeclarationNames.getCXXOperatorName(OO_Equal);
|
|
|
|
|
CXXMethodDecl *CopyAssignment =
|
|
|
|
|
CXXMethodDecl::Create(Context, ClassDecl, ClassDecl->getLocation(), Name,
|
|
|
|
|
Context.getFunctionType(RetType, &ArgType, 1,
|
|
|
|
|
false, 0),
|
2009-01-20 09:17:11 +08:00
|
|
|
|
/*isStatic=*/false, /*isInline=*/true);
|
2009-01-06 04:52:13 +08:00
|
|
|
|
CopyAssignment->setAccess(AS_public);
|
2009-01-08 03:46:03 +08:00
|
|
|
|
CopyAssignment->setImplicit();
|
2009-01-06 04:52:13 +08:00
|
|
|
|
|
|
|
|
|
// Add the parameter to the operator.
|
|
|
|
|
ParmVarDecl *FromParam = ParmVarDecl::Create(Context, CopyAssignment,
|
|
|
|
|
ClassDecl->getLocation(),
|
|
|
|
|
/*IdentifierInfo=*/0,
|
2009-01-20 09:17:11 +08:00
|
|
|
|
ArgType, VarDecl::None, 0);
|
2009-01-14 08:42:25 +08:00
|
|
|
|
CopyAssignment->setParams(Context, &FromParam, 1);
|
2009-01-06 04:52:13 +08:00
|
|
|
|
|
|
|
|
|
// Don't call addedAssignmentOperator. There is no way to distinguish an
|
|
|
|
|
// implicit from an explicit assignment operator.
|
2009-04-10 05:40:53 +08:00
|
|
|
|
ClassDecl->addDecl(Context, CopyAssignment);
|
2009-01-06 04:52:13 +08:00
|
|
|
|
}
|
|
|
|
|
|
2008-12-16 05:24:18 +08:00
|
|
|
|
if (!ClassDecl->hasUserDeclaredDestructor()) {
|
2008-11-06 04:51:48 +08:00
|
|
|
|
// C++ [class.dtor]p2:
|
|
|
|
|
// If a class has no user-declared destructor, a destructor is
|
|
|
|
|
// declared implicitly. An implicitly-declared destructor is an
|
|
|
|
|
// inline public member of its class.
|
2008-11-17 22:58:09 +08:00
|
|
|
|
DeclarationName Name
|
|
|
|
|
= Context.DeclarationNames.getCXXDestructorName(ClassType);
|
2008-11-06 04:51:48 +08:00
|
|
|
|
CXXDestructorDecl *Destructor
|
|
|
|
|
= CXXDestructorDecl::Create(Context, ClassDecl,
|
2008-11-17 22:58:09 +08:00
|
|
|
|
ClassDecl->getLocation(), Name,
|
2008-11-06 04:51:48 +08:00
|
|
|
|
Context.getFunctionType(Context.VoidTy,
|
|
|
|
|
0, 0, false, 0),
|
|
|
|
|
/*isInline=*/true,
|
|
|
|
|
/*isImplicitlyDeclared=*/true);
|
|
|
|
|
Destructor->setAccess(AS_public);
|
2009-01-08 03:46:03 +08:00
|
|
|
|
Destructor->setImplicit();
|
2009-04-10 05:40:53 +08:00
|
|
|
|
ClassDecl->addDecl(Context, Destructor);
|
2008-11-06 04:51:48 +08:00
|
|
|
|
}
|
2008-11-04 01:51:48 +08:00
|
|
|
|
}
|
|
|
|
|
|
2009-05-28 07:11:45 +08:00
|
|
|
|
void Sema::ActOnReenterTemplateScope(Scope *S, DeclPtrTy TemplateD) {
|
|
|
|
|
TemplateDecl *Template = TemplateD.getAs<TemplateDecl>();
|
|
|
|
|
if (!Template)
|
|
|
|
|
return;
|
|
|
|
|
|
|
|
|
|
TemplateParameterList *Params = Template->getTemplateParameters();
|
|
|
|
|
for (TemplateParameterList::iterator Param = Params->begin(),
|
|
|
|
|
ParamEnd = Params->end();
|
|
|
|
|
Param != ParamEnd; ++Param) {
|
|
|
|
|
NamedDecl *Named = cast<NamedDecl>(*Param);
|
|
|
|
|
if (Named->getDeclName()) {
|
|
|
|
|
S->AddDecl(DeclPtrTy::make(Named));
|
|
|
|
|
IdResolver.AddDecl(Named);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2008-12-17 05:30:33 +08:00
|
|
|
|
/// ActOnStartDelayedCXXMethodDeclaration - We have completed
|
|
|
|
|
/// parsing a top-level (non-nested) C++ class, and we are now
|
|
|
|
|
/// parsing those parts of the given Method declaration that could
|
|
|
|
|
/// not be parsed earlier (C++ [class.mem]p2), such as default
|
|
|
|
|
/// arguments. This action should enter the scope of the given
|
|
|
|
|
/// Method declaration as if we had just parsed the qualified method
|
|
|
|
|
/// name. However, it should not bring the parameters into scope;
|
|
|
|
|
/// that will be performed by ActOnDelayedCXXMethodParameter.
|
2009-03-29 03:18:32 +08:00
|
|
|
|
void Sema::ActOnStartDelayedCXXMethodDeclaration(Scope *S, DeclPtrTy MethodD) {
|
2008-12-17 05:30:33 +08:00
|
|
|
|
CXXScopeSpec SS;
|
2009-03-29 03:18:32 +08:00
|
|
|
|
FunctionDecl *Method = cast<FunctionDecl>(MethodD.getAs<Decl>());
|
2009-03-27 07:50:42 +08:00
|
|
|
|
QualType ClassTy
|
|
|
|
|
= Context.getTypeDeclType(cast<RecordDecl>(Method->getDeclContext()));
|
|
|
|
|
SS.setScopeRep(
|
|
|
|
|
NestedNameSpecifier::Create(Context, 0, false, ClassTy.getTypePtr()));
|
2008-12-17 05:30:33 +08:00
|
|
|
|
ActOnCXXEnterDeclaratorScope(S, SS);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/// ActOnDelayedCXXMethodParameter - We've already started a delayed
|
|
|
|
|
/// C++ method declaration. We're (re-)introducing the given
|
|
|
|
|
/// function parameter into scope for use in parsing later parts of
|
|
|
|
|
/// the method declaration. For example, we could see an
|
|
|
|
|
/// ActOnParamDefaultArgument event for this parameter.
|
2009-03-29 03:18:32 +08:00
|
|
|
|
void Sema::ActOnDelayedCXXMethodParameter(Scope *S, DeclPtrTy ParamD) {
|
|
|
|
|
ParmVarDecl *Param = cast<ParmVarDecl>(ParamD.getAs<Decl>());
|
2008-12-24 08:01:03 +08:00
|
|
|
|
|
|
|
|
|
// If this parameter has an unparsed default argument, clear it out
|
|
|
|
|
// to make way for the parsed default argument.
|
|
|
|
|
if (Param->hasUnparsedDefaultArg())
|
|
|
|
|
Param->setDefaultArg(0);
|
|
|
|
|
|
2009-03-29 03:18:32 +08:00
|
|
|
|
S->AddDecl(DeclPtrTy::make(Param));
|
2008-12-17 05:30:33 +08:00
|
|
|
|
if (Param->getDeclName())
|
|
|
|
|
IdResolver.AddDecl(Param);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/// ActOnFinishDelayedCXXMethodDeclaration - We have finished
|
|
|
|
|
/// processing the delayed method declaration for Method. The method
|
|
|
|
|
/// declaration is now considered finished. There may be a separate
|
|
|
|
|
/// ActOnStartOfFunctionDef action later (not necessarily
|
|
|
|
|
/// immediately!) for this method, if it was also defined inside the
|
|
|
|
|
/// class body.
|
2009-03-29 03:18:32 +08:00
|
|
|
|
void Sema::ActOnFinishDelayedCXXMethodDeclaration(Scope *S, DeclPtrTy MethodD) {
|
|
|
|
|
FunctionDecl *Method = cast<FunctionDecl>(MethodD.getAs<Decl>());
|
2008-12-17 05:30:33 +08:00
|
|
|
|
CXXScopeSpec SS;
|
2009-03-27 07:50:42 +08:00
|
|
|
|
QualType ClassTy
|
|
|
|
|
= Context.getTypeDeclType(cast<RecordDecl>(Method->getDeclContext()));
|
|
|
|
|
SS.setScopeRep(
|
|
|
|
|
NestedNameSpecifier::Create(Context, 0, false, ClassTy.getTypePtr()));
|
2008-12-17 05:30:33 +08:00
|
|
|
|
ActOnCXXExitDeclaratorScope(S, SS);
|
|
|
|
|
|
|
|
|
|
// Now that we have our default arguments, check the constructor
|
|
|
|
|
// again. It could produce additional diagnostics or affect whether
|
|
|
|
|
// the class has implicitly-declared destructors, among other
|
|
|
|
|
// things.
|
2009-04-25 16:35:12 +08:00
|
|
|
|
if (CXXConstructorDecl *Constructor = dyn_cast<CXXConstructorDecl>(Method))
|
|
|
|
|
CheckConstructor(Constructor);
|
2008-12-17 05:30:33 +08:00
|
|
|
|
|
|
|
|
|
// Check the default arguments, which we may have added.
|
|
|
|
|
if (!Method->isInvalidDecl())
|
|
|
|
|
CheckCXXDefaultArguments(Method);
|
|
|
|
|
}
|
|
|
|
|
|
2008-11-06 04:51:48 +08:00
|
|
|
|
/// CheckConstructorDeclarator - Called by ActOnDeclarator to check
|
2008-12-17 05:30:33 +08:00
|
|
|
|
/// the well-formedness of the constructor declarator @p D with type @p
|
2008-11-06 04:51:48 +08:00
|
|
|
|
/// R. If there are any errors in the declarator, this routine will
|
2009-04-25 16:28:21 +08:00
|
|
|
|
/// emit diagnostics and set the invalid bit to true. In any case, the type
|
|
|
|
|
/// will be updated to reflect a well-formed type for the constructor and
|
|
|
|
|
/// returned.
|
|
|
|
|
QualType Sema::CheckConstructorDeclarator(Declarator &D, QualType R,
|
|
|
|
|
FunctionDecl::StorageClass &SC) {
|
2008-11-06 04:51:48 +08:00
|
|
|
|
bool isVirtual = D.getDeclSpec().isVirtualSpecified();
|
|
|
|
|
|
|
|
|
|
// C++ [class.ctor]p3:
|
|
|
|
|
// A constructor shall not be virtual (10.3) or static (9.4). A
|
|
|
|
|
// constructor can be invoked for a const, volatile or const
|
|
|
|
|
// volatile object. A constructor shall not be declared const,
|
|
|
|
|
// volatile, or const volatile (9.3.2).
|
|
|
|
|
if (isVirtual) {
|
2009-04-25 16:28:21 +08:00
|
|
|
|
if (!D.isInvalidType())
|
|
|
|
|
Diag(D.getIdentifierLoc(), diag::err_constructor_cannot_be)
|
|
|
|
|
<< "virtual" << SourceRange(D.getDeclSpec().getVirtualSpecLoc())
|
|
|
|
|
<< SourceRange(D.getIdentifierLoc());
|
|
|
|
|
D.setInvalidType();
|
2008-11-06 04:51:48 +08:00
|
|
|
|
}
|
|
|
|
|
if (SC == FunctionDecl::Static) {
|
2009-04-25 16:28:21 +08:00
|
|
|
|
if (!D.isInvalidType())
|
|
|
|
|
Diag(D.getIdentifierLoc(), diag::err_constructor_cannot_be)
|
|
|
|
|
<< "static" << SourceRange(D.getDeclSpec().getStorageClassSpecLoc())
|
|
|
|
|
<< SourceRange(D.getIdentifierLoc());
|
|
|
|
|
D.setInvalidType();
|
2008-11-06 04:51:48 +08:00
|
|
|
|
SC = FunctionDecl::None;
|
|
|
|
|
}
|
2009-04-25 16:28:21 +08:00
|
|
|
|
|
|
|
|
|
DeclaratorChunk::FunctionTypeInfo &FTI = D.getTypeObject(0).Fun;
|
|
|
|
|
if (FTI.TypeQuals != 0) {
|
2008-11-06 04:51:48 +08:00
|
|
|
|
if (FTI.TypeQuals & QualType::Const)
|
2008-11-19 13:08:23 +08:00
|
|
|
|
Diag(D.getIdentifierLoc(), diag::err_invalid_qualified_constructor)
|
|
|
|
|
<< "const" << SourceRange(D.getIdentifierLoc());
|
2008-11-06 04:51:48 +08:00
|
|
|
|
if (FTI.TypeQuals & QualType::Volatile)
|
2008-11-19 13:08:23 +08:00
|
|
|
|
Diag(D.getIdentifierLoc(), diag::err_invalid_qualified_constructor)
|
|
|
|
|
<< "volatile" << SourceRange(D.getIdentifierLoc());
|
2008-11-06 04:51:48 +08:00
|
|
|
|
if (FTI.TypeQuals & QualType::Restrict)
|
2008-11-19 13:08:23 +08:00
|
|
|
|
Diag(D.getIdentifierLoc(), diag::err_invalid_qualified_constructor)
|
|
|
|
|
<< "restrict" << SourceRange(D.getIdentifierLoc());
|
2008-11-06 04:51:48 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Rebuild the function type "R" without any type qualifiers (in
|
|
|
|
|
// case any of the errors above fired) and with "void" as the
|
|
|
|
|
// return type, since constructors don't have return types. We
|
|
|
|
|
// *always* have to do this, because GetTypeForDeclarator will
|
|
|
|
|
// put in a result type of "int" when none was specified.
|
2009-02-27 07:50:07 +08:00
|
|
|
|
const FunctionProtoType *Proto = R->getAsFunctionProtoType();
|
2009-04-25 16:28:21 +08:00
|
|
|
|
return Context.getFunctionType(Context.VoidTy, Proto->arg_type_begin(),
|
|
|
|
|
Proto->getNumArgs(),
|
|
|
|
|
Proto->isVariadic(), 0);
|
2008-11-06 04:51:48 +08:00
|
|
|
|
}
|
|
|
|
|
|
2008-12-17 05:30:33 +08:00
|
|
|
|
/// CheckConstructor - Checks a fully-formed constructor for
|
|
|
|
|
/// well-formedness, issuing any diagnostics required. Returns true if
|
|
|
|
|
/// the constructor declarator is invalid.
|
2009-04-25 16:35:12 +08:00
|
|
|
|
void Sema::CheckConstructor(CXXConstructorDecl *Constructor) {
|
2009-03-27 12:38:56 +08:00
|
|
|
|
CXXRecordDecl *ClassDecl
|
|
|
|
|
= dyn_cast<CXXRecordDecl>(Constructor->getDeclContext());
|
|
|
|
|
if (!ClassDecl)
|
2009-04-25 16:35:12 +08:00
|
|
|
|
return Constructor->setInvalidDecl();
|
2008-12-17 05:30:33 +08:00
|
|
|
|
|
|
|
|
|
// C++ [class.copy]p3:
|
|
|
|
|
// A declaration of a constructor for a class X is ill-formed if
|
|
|
|
|
// its first parameter is of type (optionally cv-qualified) X and
|
|
|
|
|
// either there are no other parameters or else all other
|
|
|
|
|
// parameters have default arguments.
|
2009-03-27 12:38:56 +08:00
|
|
|
|
if (!Constructor->isInvalidDecl() &&
|
|
|
|
|
((Constructor->getNumParams() == 1) ||
|
|
|
|
|
(Constructor->getNumParams() > 1 &&
|
2009-06-06 12:14:07 +08:00
|
|
|
|
Constructor->getParamDecl(1)->hasDefaultArg()))) {
|
2008-12-17 05:30:33 +08:00
|
|
|
|
QualType ParamType = Constructor->getParamDecl(0)->getType();
|
|
|
|
|
QualType ClassTy = Context.getTagDeclType(ClassDecl);
|
|
|
|
|
if (Context.getCanonicalType(ParamType).getUnqualifiedType() == ClassTy) {
|
2009-04-02 07:51:29 +08:00
|
|
|
|
SourceLocation ParamLoc = Constructor->getParamDecl(0)->getLocation();
|
|
|
|
|
Diag(ParamLoc, diag::err_constructor_byvalue_arg)
|
2009-04-02 09:08:08 +08:00
|
|
|
|
<< CodeModificationHint::CreateInsertion(ParamLoc, " const &");
|
2009-04-25 16:35:12 +08:00
|
|
|
|
Constructor->setInvalidDecl();
|
2008-12-17 05:30:33 +08:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Notify the class that we've added a constructor.
|
|
|
|
|
ClassDecl->addedConstructor(Context, Constructor);
|
|
|
|
|
}
|
|
|
|
|
|
2009-05-01 07:18:11 +08:00
|
|
|
|
static inline bool
|
|
|
|
|
FTIHasSingleVoidArgument(DeclaratorChunk::FunctionTypeInfo &FTI) {
|
|
|
|
|
return (FTI.NumArgs == 1 && !FTI.isVariadic && FTI.ArgInfo[0].Ident == 0 &&
|
|
|
|
|
FTI.ArgInfo[0].Param &&
|
|
|
|
|
FTI.ArgInfo[0].Param.getAs<ParmVarDecl>()->getType()->isVoidType());
|
|
|
|
|
}
|
|
|
|
|
|
2008-11-06 04:51:48 +08:00
|
|
|
|
/// CheckDestructorDeclarator - Called by ActOnDeclarator to check
|
|
|
|
|
/// the well-formednes of the destructor declarator @p D with type @p
|
|
|
|
|
/// R. If there are any errors in the declarator, this routine will
|
2009-04-25 16:28:21 +08:00
|
|
|
|
/// emit diagnostics and set the declarator to invalid. Even if this happens,
|
|
|
|
|
/// will be updated to reflect a well-formed type for the destructor and
|
|
|
|
|
/// returned.
|
|
|
|
|
QualType Sema::CheckDestructorDeclarator(Declarator &D,
|
|
|
|
|
FunctionDecl::StorageClass& SC) {
|
2008-11-06 04:51:48 +08:00
|
|
|
|
// C++ [class.dtor]p1:
|
|
|
|
|
// [...] A typedef-name that names a class is a class-name
|
|
|
|
|
// (7.1.3); however, a typedef-name that names a class shall not
|
|
|
|
|
// be used as the identifier in the declarator for a destructor
|
|
|
|
|
// declaration.
|
2009-02-09 23:09:02 +08:00
|
|
|
|
QualType DeclaratorType = QualType::getFromOpaquePtr(D.getDeclaratorIdType());
|
2009-04-25 16:28:21 +08:00
|
|
|
|
if (isa<TypedefType>(DeclaratorType)) {
|
|
|
|
|
Diag(D.getIdentifierLoc(), diag::err_destructor_typedef_name)
|
2009-02-09 23:09:02 +08:00
|
|
|
|
<< DeclaratorType;
|
2009-04-25 16:28:21 +08:00
|
|
|
|
D.setInvalidType();
|
2008-11-06 04:51:48 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// C++ [class.dtor]p2:
|
|
|
|
|
// A destructor is used to destroy objects of its class type. A
|
|
|
|
|
// destructor takes no parameters, and no return type can be
|
|
|
|
|
// specified for it (not even void). The address of a destructor
|
|
|
|
|
// shall not be taken. A destructor shall not be static. A
|
|
|
|
|
// destructor can be invoked for a const, volatile or const
|
|
|
|
|
// volatile object. A destructor shall not be declared const,
|
|
|
|
|
// volatile or const volatile (9.3.2).
|
|
|
|
|
if (SC == FunctionDecl::Static) {
|
2009-04-25 16:28:21 +08:00
|
|
|
|
if (!D.isInvalidType())
|
|
|
|
|
Diag(D.getIdentifierLoc(), diag::err_destructor_cannot_be)
|
|
|
|
|
<< "static" << SourceRange(D.getDeclSpec().getStorageClassSpecLoc())
|
|
|
|
|
<< SourceRange(D.getIdentifierLoc());
|
2008-11-06 04:51:48 +08:00
|
|
|
|
SC = FunctionDecl::None;
|
2009-04-25 16:28:21 +08:00
|
|
|
|
D.setInvalidType();
|
2008-11-06 04:51:48 +08:00
|
|
|
|
}
|
2009-04-25 16:28:21 +08:00
|
|
|
|
if (D.getDeclSpec().hasTypeSpecifier() && !D.isInvalidType()) {
|
2008-11-06 04:51:48 +08:00
|
|
|
|
// Destructors don't have return types, but the parser will
|
|
|
|
|
// happily parse something like:
|
|
|
|
|
//
|
|
|
|
|
// class X {
|
|
|
|
|
// float ~X();
|
|
|
|
|
// };
|
|
|
|
|
//
|
|
|
|
|
// The return type will be eliminated later.
|
2008-11-19 13:08:23 +08:00
|
|
|
|
Diag(D.getIdentifierLoc(), diag::err_destructor_return_type)
|
|
|
|
|
<< SourceRange(D.getDeclSpec().getTypeSpecTypeLoc())
|
|
|
|
|
<< SourceRange(D.getIdentifierLoc());
|
2008-11-06 04:51:48 +08:00
|
|
|
|
}
|
2009-04-25 16:28:21 +08:00
|
|
|
|
|
|
|
|
|
DeclaratorChunk::FunctionTypeInfo &FTI = D.getTypeObject(0).Fun;
|
|
|
|
|
if (FTI.TypeQuals != 0 && !D.isInvalidType()) {
|
2008-11-06 04:51:48 +08:00
|
|
|
|
if (FTI.TypeQuals & QualType::Const)
|
2008-11-19 13:08:23 +08:00
|
|
|
|
Diag(D.getIdentifierLoc(), diag::err_invalid_qualified_destructor)
|
|
|
|
|
<< "const" << SourceRange(D.getIdentifierLoc());
|
2008-11-06 04:51:48 +08:00
|
|
|
|
if (FTI.TypeQuals & QualType::Volatile)
|
2008-11-19 13:08:23 +08:00
|
|
|
|
Diag(D.getIdentifierLoc(), diag::err_invalid_qualified_destructor)
|
|
|
|
|
<< "volatile" << SourceRange(D.getIdentifierLoc());
|
2008-11-06 04:51:48 +08:00
|
|
|
|
if (FTI.TypeQuals & QualType::Restrict)
|
2008-11-19 13:08:23 +08:00
|
|
|
|
Diag(D.getIdentifierLoc(), diag::err_invalid_qualified_destructor)
|
|
|
|
|
<< "restrict" << SourceRange(D.getIdentifierLoc());
|
2009-04-25 16:28:21 +08:00
|
|
|
|
D.setInvalidType();
|
2008-11-06 04:51:48 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Make sure we don't have any parameters.
|
2009-05-01 07:18:11 +08:00
|
|
|
|
if (FTI.NumArgs > 0 && !FTIHasSingleVoidArgument(FTI)) {
|
2008-11-06 04:51:48 +08:00
|
|
|
|
Diag(D.getIdentifierLoc(), diag::err_destructor_with_params);
|
|
|
|
|
|
|
|
|
|
// Delete the parameters.
|
2009-04-25 16:28:21 +08:00
|
|
|
|
FTI.freeArgs();
|
|
|
|
|
D.setInvalidType();
|
2008-11-06 04:51:48 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Make sure the destructor isn't variadic.
|
2009-04-25 16:28:21 +08:00
|
|
|
|
if (FTI.isVariadic) {
|
2008-11-06 04:51:48 +08:00
|
|
|
|
Diag(D.getIdentifierLoc(), diag::err_destructor_variadic);
|
2009-04-25 16:28:21 +08:00
|
|
|
|
D.setInvalidType();
|
|
|
|
|
}
|
2008-11-06 04:51:48 +08:00
|
|
|
|
|
|
|
|
|
// Rebuild the function type "R" without any type qualifiers or
|
|
|
|
|
// parameters (in case any of the errors above fired) and with
|
|
|
|
|
// "void" as the return type, since destructors don't have return
|
|
|
|
|
// types. We *always* have to do this, because GetTypeForDeclarator
|
|
|
|
|
// will put in a result type of "int" when none was specified.
|
2009-04-25 16:28:21 +08:00
|
|
|
|
return Context.getFunctionType(Context.VoidTy, 0, 0, false, 0);
|
2008-11-06 04:51:48 +08:00
|
|
|
|
}
|
|
|
|
|
|
2008-11-08 04:08:42 +08:00
|
|
|
|
/// CheckConversionDeclarator - Called by ActOnDeclarator to check the
|
|
|
|
|
/// well-formednes of the conversion function declarator @p D with
|
|
|
|
|
/// type @p R. If there are any errors in the declarator, this routine
|
|
|
|
|
/// will emit diagnostics and return true. Otherwise, it will return
|
|
|
|
|
/// false. Either way, the type @p R will be updated to reflect a
|
|
|
|
|
/// well-formed type for the conversion operator.
|
2009-04-25 16:35:12 +08:00
|
|
|
|
void Sema::CheckConversionDeclarator(Declarator &D, QualType &R,
|
2008-11-08 04:08:42 +08:00
|
|
|
|
FunctionDecl::StorageClass& SC) {
|
|
|
|
|
// C++ [class.conv.fct]p1:
|
|
|
|
|
// Neither parameter types nor return type can be specified. The
|
|
|
|
|
// type of a conversion function (8.3.5) is “function taking no
|
|
|
|
|
// parameter returning conversion-type-id.”
|
|
|
|
|
if (SC == FunctionDecl::Static) {
|
2009-04-25 16:35:12 +08:00
|
|
|
|
if (!D.isInvalidType())
|
|
|
|
|
Diag(D.getIdentifierLoc(), diag::err_conv_function_not_member)
|
|
|
|
|
<< "static" << SourceRange(D.getDeclSpec().getStorageClassSpecLoc())
|
|
|
|
|
<< SourceRange(D.getIdentifierLoc());
|
|
|
|
|
D.setInvalidType();
|
2008-11-08 04:08:42 +08:00
|
|
|
|
SC = FunctionDecl::None;
|
|
|
|
|
}
|
2009-04-25 16:35:12 +08:00
|
|
|
|
if (D.getDeclSpec().hasTypeSpecifier() && !D.isInvalidType()) {
|
2008-11-08 04:08:42 +08:00
|
|
|
|
// Conversion functions don't have return types, but the parser will
|
|
|
|
|
// happily parse something like:
|
|
|
|
|
//
|
|
|
|
|
// class X {
|
|
|
|
|
// float operator bool();
|
|
|
|
|
// };
|
|
|
|
|
//
|
|
|
|
|
// The return type will be changed later anyway.
|
2008-11-19 13:08:23 +08:00
|
|
|
|
Diag(D.getIdentifierLoc(), diag::err_conv_function_return_type)
|
|
|
|
|
<< SourceRange(D.getDeclSpec().getTypeSpecTypeLoc())
|
|
|
|
|
<< SourceRange(D.getIdentifierLoc());
|
2008-11-08 04:08:42 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Make sure we don't have any parameters.
|
2009-02-27 07:50:07 +08:00
|
|
|
|
if (R->getAsFunctionProtoType()->getNumArgs() > 0) {
|
2008-11-08 04:08:42 +08:00
|
|
|
|
Diag(D.getIdentifierLoc(), diag::err_conv_function_with_params);
|
|
|
|
|
|
|
|
|
|
// Delete the parameters.
|
2009-01-21 05:06:38 +08:00
|
|
|
|
D.getTypeObject(0).Fun.freeArgs();
|
2009-04-25 16:35:12 +08:00
|
|
|
|
D.setInvalidType();
|
2008-11-08 04:08:42 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Make sure the conversion function isn't variadic.
|
2009-04-25 16:35:12 +08:00
|
|
|
|
if (R->getAsFunctionProtoType()->isVariadic() && !D.isInvalidType()) {
|
2008-11-08 04:08:42 +08:00
|
|
|
|
Diag(D.getIdentifierLoc(), diag::err_conv_function_variadic);
|
2009-04-25 16:35:12 +08:00
|
|
|
|
D.setInvalidType();
|
|
|
|
|
}
|
2008-11-08 04:08:42 +08:00
|
|
|
|
|
|
|
|
|
// C++ [class.conv.fct]p4:
|
|
|
|
|
// The conversion-type-id shall not represent a function type nor
|
|
|
|
|
// an array type.
|
|
|
|
|
QualType ConvType = QualType::getFromOpaquePtr(D.getDeclaratorIdType());
|
|
|
|
|
if (ConvType->isArrayType()) {
|
|
|
|
|
Diag(D.getIdentifierLoc(), diag::err_conv_function_to_array);
|
|
|
|
|
ConvType = Context.getPointerType(ConvType);
|
2009-04-25 16:35:12 +08:00
|
|
|
|
D.setInvalidType();
|
2008-11-08 04:08:42 +08:00
|
|
|
|
} else if (ConvType->isFunctionType()) {
|
|
|
|
|
Diag(D.getIdentifierLoc(), diag::err_conv_function_to_function);
|
|
|
|
|
ConvType = Context.getPointerType(ConvType);
|
2009-04-25 16:35:12 +08:00
|
|
|
|
D.setInvalidType();
|
2008-11-08 04:08:42 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Rebuild the function type "R" without any parameters (in case any
|
|
|
|
|
// of the errors above fired) and with the conversion type as the
|
|
|
|
|
// return type.
|
|
|
|
|
R = Context.getFunctionType(ConvType, 0, 0, false,
|
2009-02-27 07:50:07 +08:00
|
|
|
|
R->getAsFunctionProtoType()->getTypeQuals());
|
2008-11-08 04:08:42 +08:00
|
|
|
|
|
2009-01-14 23:45:31 +08:00
|
|
|
|
// C++0x explicit conversion operators.
|
|
|
|
|
if (D.getDeclSpec().isExplicitSpecified() && !getLangOptions().CPlusPlus0x)
|
|
|
|
|
Diag(D.getDeclSpec().getExplicitSpecLoc(),
|
|
|
|
|
diag::warn_explicit_conversion_functions)
|
|
|
|
|
<< SourceRange(D.getDeclSpec().getExplicitSpecLoc());
|
2008-11-08 04:08:42 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/// ActOnConversionDeclarator - Called by ActOnDeclarator to complete
|
|
|
|
|
/// the declaration of the given C++ conversion function. This routine
|
|
|
|
|
/// is responsible for recording the conversion function in the C++
|
|
|
|
|
/// class, if possible.
|
2009-03-29 03:18:32 +08:00
|
|
|
|
Sema::DeclPtrTy Sema::ActOnConversionDeclarator(CXXConversionDecl *Conversion) {
|
2008-11-08 04:08:42 +08:00
|
|
|
|
assert(Conversion && "Expected to receive a conversion function declaration");
|
|
|
|
|
|
2008-12-12 16:25:50 +08:00
|
|
|
|
// Set the lexical context of this conversion function
|
|
|
|
|
Conversion->setLexicalDeclContext(CurContext);
|
|
|
|
|
|
|
|
|
|
CXXRecordDecl *ClassDecl = cast<CXXRecordDecl>(Conversion->getDeclContext());
|
2008-11-08 04:08:42 +08:00
|
|
|
|
|
|
|
|
|
// Make sure we aren't redeclaring the conversion function.
|
|
|
|
|
QualType ConvType = Context.getCanonicalType(Conversion->getConversionType());
|
|
|
|
|
|
|
|
|
|
// C++ [class.conv.fct]p1:
|
|
|
|
|
// [...] A conversion function is never used to convert a
|
|
|
|
|
// (possibly cv-qualified) object to the (possibly cv-qualified)
|
|
|
|
|
// same object type (or a reference to it), to a (possibly
|
|
|
|
|
// cv-qualified) base class of that type (or a reference to it),
|
|
|
|
|
// or to (possibly cv-qualified) void.
|
2009-05-16 15:39:55 +08:00
|
|
|
|
// FIXME: Suppress this warning if the conversion function ends up being a
|
|
|
|
|
// virtual function that overrides a virtual function in a base class.
|
2008-11-08 04:08:42 +08:00
|
|
|
|
QualType ClassType
|
|
|
|
|
= Context.getCanonicalType(Context.getTypeDeclType(ClassDecl));
|
|
|
|
|
if (const ReferenceType *ConvTypeRef = ConvType->getAsReferenceType())
|
|
|
|
|
ConvType = ConvTypeRef->getPointeeType();
|
|
|
|
|
if (ConvType->isRecordType()) {
|
|
|
|
|
ConvType = Context.getCanonicalType(ConvType).getUnqualifiedType();
|
|
|
|
|
if (ConvType == ClassType)
|
2008-11-20 14:13:02 +08:00
|
|
|
|
Diag(Conversion->getLocation(), diag::warn_conv_to_self_not_used)
|
2008-11-24 14:25:27 +08:00
|
|
|
|
<< ClassType;
|
2008-11-08 04:08:42 +08:00
|
|
|
|
else if (IsDerivedFrom(ClassType, ConvType))
|
2008-11-20 14:13:02 +08:00
|
|
|
|
Diag(Conversion->getLocation(), diag::warn_conv_to_base_not_used)
|
2008-11-24 14:25:27 +08:00
|
|
|
|
<< ClassType << ConvType;
|
2008-11-08 04:08:42 +08:00
|
|
|
|
} else if (ConvType->isVoidType()) {
|
2008-11-20 14:13:02 +08:00
|
|
|
|
Diag(Conversion->getLocation(), diag::warn_conv_to_void_not_used)
|
2008-11-24 14:25:27 +08:00
|
|
|
|
<< ClassType << ConvType;
|
2008-11-08 04:08:42 +08:00
|
|
|
|
}
|
|
|
|
|
|
2008-12-26 23:00:45 +08:00
|
|
|
|
if (Conversion->getPreviousDeclaration()) {
|
|
|
|
|
OverloadedFunctionDecl *Conversions = ClassDecl->getConversionFunctions();
|
|
|
|
|
for (OverloadedFunctionDecl::function_iterator
|
|
|
|
|
Conv = Conversions->function_begin(),
|
|
|
|
|
ConvEnd = Conversions->function_end();
|
|
|
|
|
Conv != ConvEnd; ++Conv) {
|
|
|
|
|
if (*Conv == Conversion->getPreviousDeclaration()) {
|
|
|
|
|
*Conv = Conversion;
|
2009-03-29 03:18:32 +08:00
|
|
|
|
return DeclPtrTy::make(Conversion);
|
2008-12-26 23:00:45 +08:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
assert(Conversion->isInvalidDecl() && "Conversion should not get here.");
|
|
|
|
|
} else
|
|
|
|
|
ClassDecl->addConversionFunction(Context, Conversion);
|
2008-11-08 04:08:42 +08:00
|
|
|
|
|
2009-03-29 03:18:32 +08:00
|
|
|
|
return DeclPtrTy::make(Conversion);
|
2008-11-08 04:08:42 +08:00
|
|
|
|
}
|
|
|
|
|
|
2008-04-27 21:50:30 +08:00
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
|
// Namespace Handling
|
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
|
|
|
|
|
|
/// ActOnStartNamespaceDef - This is called at the start of a namespace
|
|
|
|
|
/// definition.
|
2009-03-29 03:18:32 +08:00
|
|
|
|
Sema::DeclPtrTy Sema::ActOnStartNamespaceDef(Scope *NamespcScope,
|
|
|
|
|
SourceLocation IdentLoc,
|
|
|
|
|
IdentifierInfo *II,
|
|
|
|
|
SourceLocation LBrace) {
|
2008-04-27 21:50:30 +08:00
|
|
|
|
NamespaceDecl *Namespc =
|
|
|
|
|
NamespaceDecl::Create(Context, CurContext, IdentLoc, II);
|
|
|
|
|
Namespc->setLBracLoc(LBrace);
|
|
|
|
|
|
|
|
|
|
Scope *DeclRegionScope = NamespcScope->getParent();
|
|
|
|
|
|
|
|
|
|
if (II) {
|
|
|
|
|
// C++ [namespace.def]p2:
|
|
|
|
|
// The identifier in an original-namespace-definition shall not have been
|
|
|
|
|
// previously defined in the declarative region in which the
|
|
|
|
|
// original-namespace-definition appears. The identifier in an
|
|
|
|
|
// original-namespace-definition is the name of the namespace. Subsequently
|
|
|
|
|
// in that declarative region, it is treated as an original-namespace-name.
|
|
|
|
|
|
2009-02-05 01:27:36 +08:00
|
|
|
|
NamedDecl *PrevDecl = LookupName(DeclRegionScope, II, LookupOrdinaryName,
|
|
|
|
|
true);
|
2008-12-12 00:49:14 +08:00
|
|
|
|
|
|
|
|
|
if (NamespaceDecl *OrigNS = dyn_cast_or_null<NamespaceDecl>(PrevDecl)) {
|
|
|
|
|
// This is an extended namespace definition.
|
|
|
|
|
// Attach this namespace decl to the chain of extended namespace
|
|
|
|
|
// definitions.
|
|
|
|
|
OrigNS->setNextNamespace(Namespc);
|
|
|
|
|
Namespc->setOriginalNamespace(OrigNS->getOriginalNamespace());
|
|
|
|
|
|
|
|
|
|
// Remove the previous declaration from the scope.
|
2009-03-29 03:18:32 +08:00
|
|
|
|
if (DeclRegionScope->isDeclScope(DeclPtrTy::make(OrigNS))) {
|
2008-12-12 04:41:00 +08:00
|
|
|
|
IdResolver.RemoveDecl(OrigNS);
|
2009-03-29 03:18:32 +08:00
|
|
|
|
DeclRegionScope->RemoveDecl(DeclPtrTy::make(OrigNS));
|
2008-04-27 21:50:30 +08:00
|
|
|
|
}
|
2008-12-12 00:49:14 +08:00
|
|
|
|
} else if (PrevDecl) {
|
|
|
|
|
// This is an invalid name redefinition.
|
|
|
|
|
Diag(Namespc->getLocation(), diag::err_redefinition_different_kind)
|
|
|
|
|
<< Namespc->getDeclName();
|
|
|
|
|
Diag(PrevDecl->getLocation(), diag::note_previous_definition);
|
|
|
|
|
Namespc->setInvalidDecl();
|
|
|
|
|
// Continue on to push Namespc as current DeclContext and return it.
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
PushOnScopeChains(Namespc, DeclRegionScope);
|
|
|
|
|
} else {
|
2008-04-27 21:50:30 +08:00
|
|
|
|
// FIXME: Handle anonymous namespaces
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Although we could have an invalid decl (i.e. the namespace name is a
|
|
|
|
|
// redefinition), push it as current DeclContext and try to continue parsing.
|
2009-05-16 15:39:55 +08:00
|
|
|
|
// FIXME: We should be able to push Namespc here, so that the each DeclContext
|
|
|
|
|
// for the namespace has the declarations that showed up in that particular
|
|
|
|
|
// namespace definition.
|
2008-12-12 00:49:14 +08:00
|
|
|
|
PushDeclContext(NamespcScope, Namespc);
|
2009-03-29 03:18:32 +08:00
|
|
|
|
return DeclPtrTy::make(Namespc);
|
2008-04-27 21:50:30 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/// ActOnFinishNamespaceDef - This callback is called after a namespace is
|
|
|
|
|
/// exited. Decl is the DeclTy returned by ActOnStartNamespaceDef.
|
2009-03-29 03:18:32 +08:00
|
|
|
|
void Sema::ActOnFinishNamespaceDef(DeclPtrTy D, SourceLocation RBrace) {
|
|
|
|
|
Decl *Dcl = D.getAs<Decl>();
|
2008-04-27 21:50:30 +08:00
|
|
|
|
NamespaceDecl *Namespc = dyn_cast_or_null<NamespaceDecl>(Dcl);
|
|
|
|
|
assert(Namespc && "Invalid parameter, expected NamespaceDecl");
|
|
|
|
|
Namespc->setRBracLoc(RBrace);
|
|
|
|
|
PopDeclContext();
|
|
|
|
|
}
|
2008-10-07 01:10:33 +08:00
|
|
|
|
|
2009-03-29 03:18:32 +08:00
|
|
|
|
Sema::DeclPtrTy Sema::ActOnUsingDirective(Scope *S,
|
|
|
|
|
SourceLocation UsingLoc,
|
|
|
|
|
SourceLocation NamespcLoc,
|
|
|
|
|
const CXXScopeSpec &SS,
|
|
|
|
|
SourceLocation IdentLoc,
|
|
|
|
|
IdentifierInfo *NamespcName,
|
|
|
|
|
AttributeList *AttrList) {
|
2008-12-30 11:27:21 +08:00
|
|
|
|
assert(!SS.isInvalid() && "Invalid CXXScopeSpec.");
|
|
|
|
|
assert(NamespcName && "Invalid NamespcName.");
|
|
|
|
|
assert(IdentLoc.isValid() && "Invalid NamespceName location.");
|
2009-02-04 03:21:40 +08:00
|
|
|
|
assert(S->getFlags() & Scope::DeclScope && "Invalid Scope.");
|
2008-12-30 11:27:21 +08:00
|
|
|
|
|
2009-02-04 03:21:40 +08:00
|
|
|
|
UsingDirectiveDecl *UDir = 0;
|
2009-01-15 06:20:51 +08:00
|
|
|
|
|
2008-12-30 11:27:21 +08:00
|
|
|
|
// Lookup namespace name.
|
2009-02-04 03:21:40 +08:00
|
|
|
|
LookupResult R = LookupParsedName(S, &SS, NamespcName,
|
|
|
|
|
LookupNamespaceName, false);
|
|
|
|
|
if (R.isAmbiguous()) {
|
|
|
|
|
DiagnoseAmbiguousLookup(R, NamespcName, IdentLoc);
|
2009-03-29 03:18:32 +08:00
|
|
|
|
return DeclPtrTy();
|
2009-02-04 03:21:40 +08:00
|
|
|
|
}
|
2009-02-05 01:27:36 +08:00
|
|
|
|
if (NamedDecl *NS = R) {
|
2008-12-30 11:27:21 +08:00
|
|
|
|
assert(isa<NamespaceDecl>(NS) && "expected namespace decl");
|
2009-02-04 03:21:40 +08:00
|
|
|
|
// C++ [namespace.udir]p1:
|
|
|
|
|
// A using-directive specifies that the names in the nominated
|
|
|
|
|
// namespace can be used in the scope in which the
|
|
|
|
|
// using-directive appears after the using-directive. During
|
|
|
|
|
// unqualified name lookup (3.4.1), the names appear as if they
|
|
|
|
|
// were declared in the nearest enclosing namespace which
|
|
|
|
|
// contains both the using-directive and the nominated
|
|
|
|
|
// namespace. [Note: in this context, “contains” means “contains
|
|
|
|
|
// directly or indirectly”. ]
|
|
|
|
|
|
|
|
|
|
// Find enclosing context containing both using-directive and
|
|
|
|
|
// nominated namespace.
|
|
|
|
|
DeclContext *CommonAncestor = cast<DeclContext>(NS);
|
|
|
|
|
while (CommonAncestor && !CommonAncestor->Encloses(CurContext))
|
|
|
|
|
CommonAncestor = CommonAncestor->getParent();
|
|
|
|
|
|
2009-05-30 14:31:56 +08:00
|
|
|
|
UDir = UsingDirectiveDecl::Create(Context,
|
|
|
|
|
CurContext, UsingLoc,
|
|
|
|
|
NamespcLoc,
|
|
|
|
|
SS.getRange(),
|
|
|
|
|
(NestedNameSpecifier *)SS.getScopeRep(),
|
|
|
|
|
IdentLoc,
|
2009-02-04 03:21:40 +08:00
|
|
|
|
cast<NamespaceDecl>(NS),
|
|
|
|
|
CommonAncestor);
|
|
|
|
|
PushUsingDirective(S, UDir);
|
2008-12-30 11:27:21 +08:00
|
|
|
|
} else {
|
2009-01-06 15:24:29 +08:00
|
|
|
|
Diag(IdentLoc, diag::err_expected_namespace_name) << SS.getRange();
|
2008-12-30 11:27:21 +08:00
|
|
|
|
}
|
|
|
|
|
|
2009-02-04 03:21:40 +08:00
|
|
|
|
// FIXME: We ignore attributes for now.
|
2008-12-30 11:27:21 +08:00
|
|
|
|
delete AttrList;
|
2009-03-29 03:18:32 +08:00
|
|
|
|
return DeclPtrTy::make(UDir);
|
2009-02-04 03:21:40 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void Sema::PushUsingDirective(Scope *S, UsingDirectiveDecl *UDir) {
|
|
|
|
|
// If scope has associated entity, then using directive is at namespace
|
|
|
|
|
// or translation unit scope. We add UsingDirectiveDecls, into
|
|
|
|
|
// it's lookup structure.
|
|
|
|
|
if (DeclContext *Ctx = static_cast<DeclContext*>(S->getEntity()))
|
2009-04-10 05:40:53 +08:00
|
|
|
|
Ctx->addDecl(Context, UDir);
|
2009-02-04 03:21:40 +08:00
|
|
|
|
else
|
|
|
|
|
// Otherwise it is block-sope. using-directives will affect lookup
|
|
|
|
|
// only to the end of scope.
|
2009-03-29 03:18:32 +08:00
|
|
|
|
S->PushUsingDirective(DeclPtrTy::make(UDir));
|
2008-12-30 11:27:21 +08:00
|
|
|
|
}
|
2008-10-07 01:10:33 +08:00
|
|
|
|
|
2009-03-29 07:53:49 +08:00
|
|
|
|
/// getNamespaceDecl - Returns the namespace a decl represents. If the decl
|
|
|
|
|
/// is a namespace alias, returns the namespace it points to.
|
|
|
|
|
static inline NamespaceDecl *getNamespaceDecl(NamedDecl *D) {
|
|
|
|
|
if (NamespaceAliasDecl *AD = dyn_cast_or_null<NamespaceAliasDecl>(D))
|
|
|
|
|
return AD->getNamespace();
|
|
|
|
|
return dyn_cast_or_null<NamespaceDecl>(D);
|
|
|
|
|
}
|
|
|
|
|
|
2009-03-29 03:18:32 +08:00
|
|
|
|
Sema::DeclPtrTy Sema::ActOnNamespaceAliasDef(Scope *S,
|
2009-03-29 06:53:22 +08:00
|
|
|
|
SourceLocation NamespaceLoc,
|
2009-03-29 03:18:32 +08:00
|
|
|
|
SourceLocation AliasLoc,
|
|
|
|
|
IdentifierInfo *Alias,
|
|
|
|
|
const CXXScopeSpec &SS,
|
2009-03-29 06:53:22 +08:00
|
|
|
|
SourceLocation IdentLoc,
|
|
|
|
|
IdentifierInfo *Ident) {
|
2009-03-28 14:23:46 +08:00
|
|
|
|
|
2009-03-29 07:53:49 +08:00
|
|
|
|
// Lookup the namespace name.
|
|
|
|
|
LookupResult R = LookupParsedName(S, &SS, Ident, LookupNamespaceName, false);
|
|
|
|
|
|
2009-03-28 14:23:46 +08:00
|
|
|
|
// Check if we have a previous declaration with the same name.
|
2009-03-29 07:49:35 +08:00
|
|
|
|
if (NamedDecl *PrevDecl = LookupName(S, Alias, LookupOrdinaryName, true)) {
|
2009-03-29 07:53:49 +08:00
|
|
|
|
if (NamespaceAliasDecl *AD = dyn_cast<NamespaceAliasDecl>(PrevDecl)) {
|
|
|
|
|
// We already have an alias with the same name that points to the same
|
|
|
|
|
// namespace, so don't create a new one.
|
|
|
|
|
if (!R.isAmbiguous() && AD->getNamespace() == getNamespaceDecl(R))
|
|
|
|
|
return DeclPtrTy();
|
|
|
|
|
}
|
|
|
|
|
|
2009-03-28 14:23:46 +08:00
|
|
|
|
unsigned DiagID = isa<NamespaceDecl>(PrevDecl) ? diag::err_redefinition :
|
|
|
|
|
diag::err_redefinition_different_kind;
|
|
|
|
|
Diag(AliasLoc, DiagID) << Alias;
|
|
|
|
|
Diag(PrevDecl->getLocation(), diag::note_previous_definition);
|
2009-03-29 03:18:32 +08:00
|
|
|
|
return DeclPtrTy();
|
2009-03-28 14:23:46 +08:00
|
|
|
|
}
|
|
|
|
|
|
2009-03-28 14:42:02 +08:00
|
|
|
|
if (R.isAmbiguous()) {
|
2009-03-29 06:53:22 +08:00
|
|
|
|
DiagnoseAmbiguousLookup(R, Ident, IdentLoc);
|
2009-03-29 03:18:32 +08:00
|
|
|
|
return DeclPtrTy();
|
2009-03-28 14:42:02 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (!R) {
|
|
|
|
|
Diag(NamespaceLoc, diag::err_expected_namespace_name) << SS.getRange();
|
2009-03-29 03:18:32 +08:00
|
|
|
|
return DeclPtrTy();
|
2009-03-28 14:42:02 +08:00
|
|
|
|
}
|
|
|
|
|
|
2009-03-29 06:58:02 +08:00
|
|
|
|
NamespaceAliasDecl *AliasDecl =
|
2009-05-30 14:48:27 +08:00
|
|
|
|
NamespaceAliasDecl::Create(Context, CurContext, NamespaceLoc, AliasLoc,
|
|
|
|
|
Alias, SS.getRange(),
|
|
|
|
|
(NestedNameSpecifier *)SS.getScopeRep(),
|
2009-03-29 06:58:02 +08:00
|
|
|
|
IdentLoc, R);
|
|
|
|
|
|
2009-04-10 05:40:53 +08:00
|
|
|
|
CurContext->addDecl(Context, AliasDecl);
|
2009-03-29 06:58:02 +08:00
|
|
|
|
return DeclPtrTy::make(AliasDecl);
|
2009-03-28 13:27:17 +08:00
|
|
|
|
}
|
|
|
|
|
|
2009-04-17 07:50:50 +08:00
|
|
|
|
void Sema::InitializeVarWithConstructor(VarDecl *VD,
|
|
|
|
|
CXXConstructorDecl *Constructor,
|
|
|
|
|
QualType DeclInitType,
|
|
|
|
|
Expr **Exprs, unsigned NumExprs) {
|
2009-05-31 04:56:46 +08:00
|
|
|
|
Expr *Temp = CXXConstructExpr::Create(Context, DeclInitType, Constructor,
|
2009-04-24 13:16:06 +08:00
|
|
|
|
false, Exprs, NumExprs);
|
2009-05-27 02:54:04 +08:00
|
|
|
|
VD->setInit(Context, Temp);
|
2009-04-17 07:50:50 +08:00
|
|
|
|
}
|
|
|
|
|
|
2008-10-07 01:10:33 +08:00
|
|
|
|
/// AddCXXDirectInitializerToDecl - This action is called immediately after
|
|
|
|
|
/// ActOnDeclarator, when a C++ direct initializer is present.
|
|
|
|
|
/// e.g: "int x(1);"
|
2009-03-29 03:18:32 +08:00
|
|
|
|
void Sema::AddCXXDirectInitializerToDecl(DeclPtrTy Dcl,
|
|
|
|
|
SourceLocation LParenLoc,
|
2009-03-16 01:47:39 +08:00
|
|
|
|
MultiExprArg Exprs,
|
2008-10-07 01:10:33 +08:00
|
|
|
|
SourceLocation *CommaLocs,
|
|
|
|
|
SourceLocation RParenLoc) {
|
2009-03-16 01:47:39 +08:00
|
|
|
|
unsigned NumExprs = Exprs.size();
|
|
|
|
|
assert(NumExprs != 0 && Exprs.get() && "missing expressions");
|
2009-03-29 03:18:32 +08:00
|
|
|
|
Decl *RealDecl = Dcl.getAs<Decl>();
|
2008-10-07 01:10:33 +08:00
|
|
|
|
|
|
|
|
|
// If there is no declaration, there was an error parsing it. Just ignore
|
|
|
|
|
// the initializer.
|
2009-03-29 03:18:32 +08:00
|
|
|
|
if (RealDecl == 0)
|
2008-10-07 01:10:33 +08:00
|
|
|
|
return;
|
|
|
|
|
|
|
|
|
|
VarDecl *VDecl = dyn_cast<VarDecl>(RealDecl);
|
|
|
|
|
if (!VDecl) {
|
|
|
|
|
Diag(RealDecl->getLocation(), diag::err_illegal_initializer);
|
|
|
|
|
RealDecl->setInvalidDecl();
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
2009-03-25 00:43:20 +08:00
|
|
|
|
// FIXME: Need to handle dependent types and expressions here.
|
|
|
|
|
|
2008-10-07 07:08:37 +08:00
|
|
|
|
// We will treat direct-initialization as a copy-initialization:
|
|
|
|
|
// int x(1); -as-> int x = 1;
|
2008-10-07 01:10:33 +08:00
|
|
|
|
// ClassType x(a,b,c); -as-> ClassType x = ClassType(a,b,c);
|
|
|
|
|
//
|
|
|
|
|
// Clients that want to distinguish between the two forms, can check for
|
|
|
|
|
// direct initializer using VarDecl::hasCXXDirectInitializer().
|
|
|
|
|
// A major benefit is that clients that don't particularly care about which
|
|
|
|
|
// exactly form was it (like the CodeGen) can handle both cases without
|
|
|
|
|
// special case code.
|
2008-10-07 02:37:09 +08:00
|
|
|
|
|
2008-10-07 01:10:33 +08:00
|
|
|
|
// C++ 8.5p11:
|
|
|
|
|
// The form of initialization (using parentheses or '=') is generally
|
|
|
|
|
// insignificant, but does matter when the entity being initialized has a
|
2008-10-07 02:37:09 +08:00
|
|
|
|
// class type.
|
2008-11-04 04:45:27 +08:00
|
|
|
|
QualType DeclInitType = VDecl->getType();
|
|
|
|
|
if (const ArrayType *Array = Context.getAsArrayType(DeclInitType))
|
|
|
|
|
DeclInitType = Array->getElementType();
|
2008-10-07 02:37:09 +08:00
|
|
|
|
|
2009-03-25 00:43:20 +08:00
|
|
|
|
// FIXME: This isn't the right place to complete the type.
|
|
|
|
|
if (RequireCompleteType(VDecl->getLocation(), VDecl->getType(),
|
|
|
|
|
diag::err_typecheck_decl_incomplete_type)) {
|
|
|
|
|
VDecl->setInvalidDecl();
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
2008-10-07 02:37:09 +08:00
|
|
|
|
if (VDecl->getType()->isRecordType()) {
|
2008-11-04 04:45:27 +08:00
|
|
|
|
CXXConstructorDecl *Constructor
|
2009-03-16 01:47:39 +08:00
|
|
|
|
= PerformInitializationByConstructor(DeclInitType,
|
|
|
|
|
(Expr **)Exprs.get(), NumExprs,
|
2008-11-05 23:29:30 +08:00
|
|
|
|
VDecl->getLocation(),
|
|
|
|
|
SourceRange(VDecl->getLocation(),
|
|
|
|
|
RParenLoc),
|
2008-11-24 13:29:24 +08:00
|
|
|
|
VDecl->getDeclName(),
|
2008-11-05 23:29:30 +08:00
|
|
|
|
IK_Direct);
|
2009-03-16 01:47:39 +08:00
|
|
|
|
if (!Constructor)
|
2008-11-04 04:45:27 +08:00
|
|
|
|
RealDecl->setInvalidDecl();
|
2009-04-16 05:48:18 +08:00
|
|
|
|
else {
|
|
|
|
|
VDecl->setCXXDirectInitializer(true);
|
2009-04-17 07:50:50 +08:00
|
|
|
|
InitializeVarWithConstructor(VDecl, Constructor, DeclInitType,
|
|
|
|
|
(Expr**)Exprs.release(), NumExprs);
|
2009-04-16 05:48:18 +08:00
|
|
|
|
}
|
2008-10-07 02:37:09 +08:00
|
|
|
|
return;
|
|
|
|
|
}
|
2008-10-07 01:10:33 +08:00
|
|
|
|
|
2008-10-07 07:08:37 +08:00
|
|
|
|
if (NumExprs > 1) {
|
2008-11-19 13:27:50 +08:00
|
|
|
|
Diag(CommaLocs[0], diag::err_builtin_direct_init_more_than_one_arg)
|
|
|
|
|
<< SourceRange(VDecl->getLocation(), RParenLoc);
|
2008-10-07 01:10:33 +08:00
|
|
|
|
RealDecl->setInvalidDecl();
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Let clients know that initialization was done with a direct initializer.
|
|
|
|
|
VDecl->setCXXDirectInitializer(true);
|
2008-10-07 07:08:37 +08:00
|
|
|
|
|
|
|
|
|
assert(NumExprs == 1 && "Expected 1 expression");
|
|
|
|
|
// Set the init expression, handles conversions.
|
2009-03-16 01:47:39 +08:00
|
|
|
|
AddInitializerToDecl(Dcl, ExprArg(*this, Exprs.release()[0]),
|
|
|
|
|
/*DirectInit=*/true);
|
2008-10-07 01:10:33 +08:00
|
|
|
|
}
|
2008-10-29 08:13:59 +08:00
|
|
|
|
|
2008-11-05 23:29:30 +08:00
|
|
|
|
/// PerformInitializationByConstructor - Perform initialization by
|
|
|
|
|
/// constructor (C++ [dcl.init]p14), which may occur as part of
|
|
|
|
|
/// direct-initialization or copy-initialization. We are initializing
|
|
|
|
|
/// an object of type @p ClassType with the given arguments @p
|
|
|
|
|
/// Args. @p Loc is the location in the source code where the
|
|
|
|
|
/// initializer occurs (e.g., a declaration, member initializer,
|
|
|
|
|
/// functional cast, etc.) while @p Range covers the whole
|
|
|
|
|
/// initialization. @p InitEntity is the entity being initialized,
|
|
|
|
|
/// which may by the name of a declaration or a type. @p Kind is the
|
|
|
|
|
/// kind of initialization we're performing, which affects whether
|
|
|
|
|
/// explicit constructors will be considered. When successful, returns
|
2008-11-04 04:45:27 +08:00
|
|
|
|
/// the constructor that will be used to perform the initialization;
|
2008-11-05 23:29:30 +08:00
|
|
|
|
/// when the initialization fails, emits a diagnostic and returns
|
|
|
|
|
/// null.
|
2008-11-04 04:45:27 +08:00
|
|
|
|
CXXConstructorDecl *
|
2008-11-05 23:29:30 +08:00
|
|
|
|
Sema::PerformInitializationByConstructor(QualType ClassType,
|
|
|
|
|
Expr **Args, unsigned NumArgs,
|
|
|
|
|
SourceLocation Loc, SourceRange Range,
|
2008-11-24 13:29:24 +08:00
|
|
|
|
DeclarationName InitEntity,
|
2008-11-05 23:29:30 +08:00
|
|
|
|
InitializationKind Kind) {
|
2008-11-04 04:45:27 +08:00
|
|
|
|
const RecordType *ClassRec = ClassType->getAsRecordType();
|
|
|
|
|
assert(ClassRec && "Can only initialize a class type here");
|
|
|
|
|
|
|
|
|
|
// C++ [dcl.init]p14:
|
|
|
|
|
//
|
|
|
|
|
// If the initialization is direct-initialization, or if it is
|
|
|
|
|
// copy-initialization where the cv-unqualified version of the
|
|
|
|
|
// source type is the same class as, or a derived class of, the
|
|
|
|
|
// class of the destination, constructors are considered. The
|
|
|
|
|
// applicable constructors are enumerated (13.3.1.3), and the
|
|
|
|
|
// best one is chosen through overload resolution (13.3). The
|
|
|
|
|
// constructor so selected is called to initialize the object,
|
|
|
|
|
// with the initializer expression(s) as its argument(s). If no
|
|
|
|
|
// constructor applies, or the overload resolution is ambiguous,
|
|
|
|
|
// the initialization is ill-formed.
|
|
|
|
|
const CXXRecordDecl *ClassDecl = cast<CXXRecordDecl>(ClassRec->getDecl());
|
|
|
|
|
OverloadCandidateSet CandidateSet;
|
2008-11-05 23:29:30 +08:00
|
|
|
|
|
|
|
|
|
// Add constructors to the overload set.
|
2008-12-16 05:24:18 +08:00
|
|
|
|
DeclarationName ConstructorName
|
|
|
|
|
= Context.DeclarationNames.getCXXConstructorName(
|
|
|
|
|
Context.getCanonicalType(ClassType.getUnqualifiedType()));
|
2008-12-23 08:26:44 +08:00
|
|
|
|
DeclContext::lookup_const_iterator Con, ConEnd;
|
2009-04-10 05:40:53 +08:00
|
|
|
|
for (llvm::tie(Con, ConEnd) = ClassDecl->lookup(Context, ConstructorName);
|
2008-12-23 08:26:44 +08:00
|
|
|
|
Con != ConEnd; ++Con) {
|
|
|
|
|
CXXConstructorDecl *Constructor = cast<CXXConstructorDecl>(*Con);
|
2008-11-05 23:29:30 +08:00
|
|
|
|
if ((Kind == IK_Direct) ||
|
|
|
|
|
(Kind == IK_Copy && Constructor->isConvertingConstructor()) ||
|
|
|
|
|
(Kind == IK_Default && Constructor->isDefaultConstructor()))
|
|
|
|
|
AddOverloadCandidate(Constructor, Args, NumArgs, CandidateSet);
|
|
|
|
|
}
|
|
|
|
|
|
2008-12-16 05:24:18 +08:00
|
|
|
|
// FIXME: When we decide not to synthesize the implicitly-declared
|
|
|
|
|
// constructors, we'll need to make them appear here.
|
|
|
|
|
|
2008-11-04 04:45:27 +08:00
|
|
|
|
OverloadCandidateSet::iterator Best;
|
|
|
|
|
switch (BestViableFunction(CandidateSet, Best)) {
|
|
|
|
|
case OR_Success:
|
|
|
|
|
// We found a constructor. Return it.
|
|
|
|
|
return cast<CXXConstructorDecl>(Best->Function);
|
|
|
|
|
|
|
|
|
|
case OR_No_Viable_Function:
|
2009-02-03 01:43:21 +08:00
|
|
|
|
if (InitEntity)
|
|
|
|
|
Diag(Loc, diag::err_ovl_no_viable_function_in_init)
|
2009-02-17 15:29:20 +08:00
|
|
|
|
<< InitEntity << Range;
|
2009-02-03 01:43:21 +08:00
|
|
|
|
else
|
|
|
|
|
Diag(Loc, diag::err_ovl_no_viable_function_in_init)
|
2009-02-17 15:29:20 +08:00
|
|
|
|
<< ClassType << Range;
|
2008-11-22 21:44:36 +08:00
|
|
|
|
PrintOverloadCandidates(CandidateSet, /*OnlyViable=*/false);
|
2008-11-04 04:45:27 +08:00
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
|
|
case OR_Ambiguous:
|
2009-02-03 01:43:21 +08:00
|
|
|
|
if (InitEntity)
|
|
|
|
|
Diag(Loc, diag::err_ovl_ambiguous_init) << InitEntity << Range;
|
|
|
|
|
else
|
|
|
|
|
Diag(Loc, diag::err_ovl_ambiguous_init) << ClassType << Range;
|
2008-11-04 04:45:27 +08:00
|
|
|
|
PrintOverloadCandidates(CandidateSet, /*OnlyViable=*/true);
|
|
|
|
|
return 0;
|
2009-02-19 05:56:37 +08:00
|
|
|
|
|
|
|
|
|
case OR_Deleted:
|
|
|
|
|
if (InitEntity)
|
|
|
|
|
Diag(Loc, diag::err_ovl_deleted_init)
|
|
|
|
|
<< Best->Function->isDeleted()
|
|
|
|
|
<< InitEntity << Range;
|
|
|
|
|
else
|
|
|
|
|
Diag(Loc, diag::err_ovl_deleted_init)
|
|
|
|
|
<< Best->Function->isDeleted()
|
|
|
|
|
<< InitEntity << Range;
|
|
|
|
|
PrintOverloadCandidates(CandidateSet, /*OnlyViable=*/true);
|
|
|
|
|
return 0;
|
2008-11-04 04:45:27 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
|
2008-10-29 08:13:59 +08:00
|
|
|
|
/// CompareReferenceRelationship - Compare the two types T1 and T2 to
|
|
|
|
|
/// determine whether they are reference-related,
|
|
|
|
|
/// reference-compatible, reference-compatible with added
|
|
|
|
|
/// qualification, or incompatible, for use in C++ initialization by
|
|
|
|
|
/// reference (C++ [dcl.ref.init]p4). Neither type can be a reference
|
|
|
|
|
/// type, and the first type (T1) is the pointee type of the reference
|
|
|
|
|
/// type being initialized.
|
|
|
|
|
Sema::ReferenceCompareResult
|
2008-10-29 10:00:59 +08:00
|
|
|
|
Sema::CompareReferenceRelationship(QualType T1, QualType T2,
|
|
|
|
|
bool& DerivedToBase) {
|
2009-03-17 07:22:08 +08:00
|
|
|
|
assert(!T1->isReferenceType() &&
|
|
|
|
|
"T1 must be the pointee type of the reference type");
|
2008-10-29 08:13:59 +08:00
|
|
|
|
assert(!T2->isReferenceType() && "T2 cannot be a reference type");
|
|
|
|
|
|
|
|
|
|
T1 = Context.getCanonicalType(T1);
|
|
|
|
|
T2 = Context.getCanonicalType(T2);
|
|
|
|
|
QualType UnqualT1 = T1.getUnqualifiedType();
|
|
|
|
|
QualType UnqualT2 = T2.getUnqualifiedType();
|
|
|
|
|
|
|
|
|
|
// C++ [dcl.init.ref]p4:
|
|
|
|
|
// Given types “cv1 T1” and “cv2 T2,” “cv1 T1” is
|
|
|
|
|
// reference-related to “cv2 T2” if T1 is the same type as T2, or
|
|
|
|
|
// T1 is a base class of T2.
|
2008-10-29 10:00:59 +08:00
|
|
|
|
if (UnqualT1 == UnqualT2)
|
|
|
|
|
DerivedToBase = false;
|
|
|
|
|
else if (IsDerivedFrom(UnqualT2, UnqualT1))
|
|
|
|
|
DerivedToBase = true;
|
|
|
|
|
else
|
2008-10-29 08:13:59 +08:00
|
|
|
|
return Ref_Incompatible;
|
|
|
|
|
|
|
|
|
|
// At this point, we know that T1 and T2 are reference-related (at
|
|
|
|
|
// least).
|
|
|
|
|
|
|
|
|
|
// C++ [dcl.init.ref]p4:
|
|
|
|
|
// "cv1 T1” is reference-compatible with “cv2 T2” if T1 is
|
|
|
|
|
// reference-related to T2 and cv1 is the same cv-qualification
|
|
|
|
|
// as, or greater cv-qualification than, cv2. For purposes of
|
|
|
|
|
// overload resolution, cases for which cv1 is greater
|
|
|
|
|
// cv-qualification than cv2 are identified as
|
|
|
|
|
// reference-compatible with added qualification (see 13.3.3.2).
|
|
|
|
|
if (T1.getCVRQualifiers() == T2.getCVRQualifiers())
|
|
|
|
|
return Ref_Compatible;
|
|
|
|
|
else if (T1.isMoreQualifiedThan(T2))
|
|
|
|
|
return Ref_Compatible_With_Added_Qualification;
|
|
|
|
|
else
|
|
|
|
|
return Ref_Related;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/// CheckReferenceInit - Check the initialization of a reference
|
|
|
|
|
/// variable with the given initializer (C++ [dcl.init.ref]). Init is
|
|
|
|
|
/// the initializer (either a simple initializer or an initializer
|
2008-10-30 07:31:03 +08:00
|
|
|
|
/// list), and DeclType is the type of the declaration. When ICS is
|
|
|
|
|
/// non-null, this routine will compute the implicit conversion
|
|
|
|
|
/// sequence according to C++ [over.ics.ref] and will not produce any
|
|
|
|
|
/// diagnostics; when ICS is null, it will emit diagnostics when any
|
|
|
|
|
/// errors are found. Either way, a return value of true indicates
|
|
|
|
|
/// that there was a failure, a return value of false indicates that
|
|
|
|
|
/// the reference initialization succeeded.
|
2008-11-04 03:09:14 +08:00
|
|
|
|
///
|
|
|
|
|
/// When @p SuppressUserConversions, user-defined conversions are
|
|
|
|
|
/// suppressed.
|
2009-01-14 23:45:31 +08:00
|
|
|
|
/// When @p AllowExplicit, we also permit explicit user-defined
|
|
|
|
|
/// conversion functions.
|
2009-04-13 01:16:29 +08:00
|
|
|
|
/// When @p ForceRValue, we unconditionally treat the initializer as an rvalue.
|
2008-10-29 10:00:59 +08:00
|
|
|
|
bool
|
2009-04-17 01:51:27 +08:00
|
|
|
|
Sema::CheckReferenceInit(Expr *&Init, QualType DeclType,
|
2008-11-04 03:09:14 +08:00
|
|
|
|
ImplicitConversionSequence *ICS,
|
2009-01-14 23:45:31 +08:00
|
|
|
|
bool SuppressUserConversions,
|
2009-04-13 01:16:29 +08:00
|
|
|
|
bool AllowExplicit, bool ForceRValue) {
|
2008-10-29 08:13:59 +08:00
|
|
|
|
assert(DeclType->isReferenceType() && "Reference init needs a reference");
|
|
|
|
|
|
|
|
|
|
QualType T1 = DeclType->getAsReferenceType()->getPointeeType();
|
|
|
|
|
QualType T2 = Init->getType();
|
|
|
|
|
|
2008-11-11 04:40:00 +08:00
|
|
|
|
// If the initializer is the address of an overloaded function, try
|
|
|
|
|
// to resolve the overloaded function. If all goes well, T2 is the
|
|
|
|
|
// type of the resulting function.
|
2009-03-14 02:40:31 +08:00
|
|
|
|
if (Context.getCanonicalType(T2) == Context.OverloadTy) {
|
2008-11-11 04:40:00 +08:00
|
|
|
|
FunctionDecl *Fn = ResolveAddressOfOverloadedFunction(Init, DeclType,
|
|
|
|
|
ICS != 0);
|
|
|
|
|
if (Fn) {
|
|
|
|
|
// Since we're performing this reference-initialization for
|
|
|
|
|
// real, update the initializer with the resulting function.
|
2009-02-19 05:56:37 +08:00
|
|
|
|
if (!ICS) {
|
|
|
|
|
if (DiagnoseUseOfDecl(Fn, Init->getSourceRange().getBegin()))
|
|
|
|
|
return true;
|
|
|
|
|
|
2008-11-11 04:40:00 +08:00
|
|
|
|
FixOverloadedFunctionReference(Init, Fn);
|
2009-02-19 05:56:37 +08:00
|
|
|
|
}
|
2008-11-11 04:40:00 +08:00
|
|
|
|
|
|
|
|
|
T2 = Fn->getType();
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2008-10-29 10:00:59 +08:00
|
|
|
|
// Compute some basic properties of the types and the initializer.
|
2009-03-17 07:22:08 +08:00
|
|
|
|
bool isRValRef = DeclType->isRValueReferenceType();
|
2008-10-29 10:00:59 +08:00
|
|
|
|
bool DerivedToBase = false;
|
2009-04-13 01:16:29 +08:00
|
|
|
|
Expr::isLvalueResult InitLvalue = ForceRValue ? Expr::LV_InvalidExpression :
|
|
|
|
|
Init->isLvalue(Context);
|
2008-10-29 10:00:59 +08:00
|
|
|
|
ReferenceCompareResult RefRelationship
|
|
|
|
|
= CompareReferenceRelationship(T1, T2, DerivedToBase);
|
|
|
|
|
|
|
|
|
|
// Most paths end in a failed conversion.
|
|
|
|
|
if (ICS)
|
|
|
|
|
ICS->ConversionKind = ImplicitConversionSequence::BadConversion;
|
2008-10-29 08:13:59 +08:00
|
|
|
|
|
|
|
|
|
// C++ [dcl.init.ref]p5:
|
|
|
|
|
// A reference to type “cv1 T1” is initialized by an expression
|
|
|
|
|
// of type “cv2 T2” as follows:
|
|
|
|
|
|
|
|
|
|
// -- If the initializer expression
|
|
|
|
|
|
2009-03-29 23:27:50 +08:00
|
|
|
|
// Rvalue references cannot bind to lvalues (N2812).
|
|
|
|
|
// There is absolutely no situation where they can. In particular, note that
|
|
|
|
|
// this is ill-formed, even if B has a user-defined conversion to A&&:
|
|
|
|
|
// B b;
|
|
|
|
|
// A&& r = b;
|
|
|
|
|
if (isRValRef && InitLvalue == Expr::LV_Valid) {
|
|
|
|
|
if (!ICS)
|
|
|
|
|
Diag(Init->getSourceRange().getBegin(), diag::err_lvalue_to_rvalue_ref)
|
|
|
|
|
<< Init->getSourceRange();
|
|
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
|
2008-10-29 08:13:59 +08:00
|
|
|
|
bool BindsDirectly = false;
|
|
|
|
|
// -- is an lvalue (but is not a bit-field), and “cv1 T1” is
|
|
|
|
|
// reference-compatible with “cv2 T2,” or
|
2008-10-29 10:00:59 +08:00
|
|
|
|
//
|
|
|
|
|
// Note that the bit-field check is skipped if we are just computing
|
|
|
|
|
// the implicit conversion sequence (C++ [over.best.ics]p2).
|
2009-05-02 10:18:30 +08:00
|
|
|
|
if (InitLvalue == Expr::LV_Valid && (ICS || !Init->getBitField()) &&
|
2008-10-29 10:00:59 +08:00
|
|
|
|
RefRelationship >= Ref_Compatible_With_Added_Qualification) {
|
2008-10-29 08:13:59 +08:00
|
|
|
|
BindsDirectly = true;
|
|
|
|
|
|
2008-10-29 10:00:59 +08:00
|
|
|
|
if (ICS) {
|
|
|
|
|
// C++ [over.ics.ref]p1:
|
|
|
|
|
// When a parameter of reference type binds directly (8.5.3)
|
|
|
|
|
// to an argument expression, the implicit conversion sequence
|
|
|
|
|
// is the identity conversion, unless the argument expression
|
|
|
|
|
// has a type that is a derived class of the parameter type,
|
|
|
|
|
// in which case the implicit conversion sequence is a
|
|
|
|
|
// derived-to-base Conversion (13.3.3.1).
|
|
|
|
|
ICS->ConversionKind = ImplicitConversionSequence::StandardConversion;
|
|
|
|
|
ICS->Standard.First = ICK_Identity;
|
|
|
|
|
ICS->Standard.Second = DerivedToBase? ICK_Derived_To_Base : ICK_Identity;
|
|
|
|
|
ICS->Standard.Third = ICK_Identity;
|
|
|
|
|
ICS->Standard.FromTypePtr = T2.getAsOpaquePtr();
|
|
|
|
|
ICS->Standard.ToTypePtr = T1.getAsOpaquePtr();
|
2008-10-29 22:50:44 +08:00
|
|
|
|
ICS->Standard.ReferenceBinding = true;
|
|
|
|
|
ICS->Standard.DirectBinding = true;
|
2009-03-29 23:27:50 +08:00
|
|
|
|
ICS->Standard.RRefBinding = false;
|
2009-04-18 00:30:52 +08:00
|
|
|
|
ICS->Standard.CopyConstructor = 0;
|
2008-10-29 10:00:59 +08:00
|
|
|
|
|
|
|
|
|
// Nothing more to do: the inaccessibility/ambiguity check for
|
|
|
|
|
// derived-to-base conversions is suppressed when we're
|
|
|
|
|
// computing the implicit conversion sequence (C++
|
|
|
|
|
// [over.best.ics]p2).
|
|
|
|
|
return false;
|
|
|
|
|
} else {
|
|
|
|
|
// Perform the conversion.
|
2009-05-16 15:39:55 +08:00
|
|
|
|
// FIXME: Binding to a subobject of the lvalue is going to require more
|
|
|
|
|
// AST annotation than this.
|
Implement support for operator overloading using candidate operator
functions for built-in operators, e.g., the builtin
bool operator==(int const*, int const*)
can be used for the expression "x1 == x2" given:
struct X {
operator int const*();
} x1, x2;
The scheme for handling these built-in operators is relatively simple:
for each candidate required by the standard, create a special kind of
candidate function for the built-in. If overload resolution picks the
built-in operator, we perform the appropriate conversions on the
arguments and then let the normal built-in operator take care of it.
There may be some optimization opportunity left: if we can reduce the
number of built-in operator overloads we generate, overload resolution
for these cases will go faster. However, one must be careful when
doing this: GCC generates too few operator overloads in our little
test program, and fails to compile it because none of the overloads it
generates match.
Note that we only support operator overload for non-member binary
operators at the moment. The other operators will follow.
As part of this change, ImplicitCastExpr can now be an lvalue.
llvm-svn: 59148
2008-11-13 01:17:38 +08:00
|
|
|
|
ImpCastExprToType(Init, T1, /*isLvalue=*/true);
|
2008-10-29 08:13:59 +08:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// -- has a class type (i.e., T2 is a class type) and can be
|
|
|
|
|
// implicitly converted to an lvalue of type “cv3 T3,”
|
|
|
|
|
// where “cv1 T1” is reference-compatible with “cv3 T3”
|
|
|
|
|
// 92) (this conversion is selected by enumerating the
|
|
|
|
|
// applicable conversion functions (13.3.1.6) and choosing
|
|
|
|
|
// the best one through overload resolution (13.3)),
|
2009-03-17 07:22:08 +08:00
|
|
|
|
if (!isRValRef && !SuppressUserConversions && T2->isRecordType()) {
|
2008-11-11 00:14:15 +08:00
|
|
|
|
// FIXME: Look for conversions in base classes!
|
|
|
|
|
CXXRecordDecl *T2RecordDecl
|
|
|
|
|
= dyn_cast<CXXRecordDecl>(T2->getAsRecordType()->getDecl());
|
|
|
|
|
|
|
|
|
|
OverloadCandidateSet CandidateSet;
|
|
|
|
|
OverloadedFunctionDecl *Conversions
|
|
|
|
|
= T2RecordDecl->getConversionFunctions();
|
|
|
|
|
for (OverloadedFunctionDecl::function_iterator Func
|
|
|
|
|
= Conversions->function_begin();
|
|
|
|
|
Func != Conversions->function_end(); ++Func) {
|
|
|
|
|
CXXConversionDecl *Conv = cast<CXXConversionDecl>(*Func);
|
2009-03-23 05:28:55 +08:00
|
|
|
|
|
2008-11-11 00:14:15 +08:00
|
|
|
|
// If the conversion function doesn't return a reference type,
|
|
|
|
|
// it can't be considered for this conversion.
|
2009-03-17 07:22:08 +08:00
|
|
|
|
if (Conv->getConversionType()->isLValueReferenceType() &&
|
2009-01-14 23:45:31 +08:00
|
|
|
|
(AllowExplicit || !Conv->isExplicit()))
|
2008-11-11 00:14:15 +08:00
|
|
|
|
AddConversionCandidate(Conv, Init, DeclType, CandidateSet);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
OverloadCandidateSet::iterator Best;
|
|
|
|
|
switch (BestViableFunction(CandidateSet, Best)) {
|
|
|
|
|
case OR_Success:
|
|
|
|
|
// This is a direct binding.
|
|
|
|
|
BindsDirectly = true;
|
|
|
|
|
|
|
|
|
|
if (ICS) {
|
|
|
|
|
// C++ [over.ics.ref]p1:
|
|
|
|
|
//
|
|
|
|
|
// [...] If the parameter binds directly to the result of
|
|
|
|
|
// applying a conversion function to the argument
|
|
|
|
|
// expression, the implicit conversion sequence is a
|
|
|
|
|
// user-defined conversion sequence (13.3.3.1.2), with the
|
|
|
|
|
// second standard conversion sequence either an identity
|
|
|
|
|
// conversion or, if the conversion function returns an
|
|
|
|
|
// entity of a type that is a derived class of the parameter
|
|
|
|
|
// type, a derived-to-base Conversion.
|
|
|
|
|
ICS->ConversionKind = ImplicitConversionSequence::UserDefinedConversion;
|
|
|
|
|
ICS->UserDefined.Before = Best->Conversions[0].Standard;
|
|
|
|
|
ICS->UserDefined.After = Best->FinalConversion;
|
|
|
|
|
ICS->UserDefined.ConversionFunction = Best->Function;
|
|
|
|
|
assert(ICS->UserDefined.After.ReferenceBinding &&
|
|
|
|
|
ICS->UserDefined.After.DirectBinding &&
|
|
|
|
|
"Expected a direct reference binding!");
|
|
|
|
|
return false;
|
|
|
|
|
} else {
|
|
|
|
|
// Perform the conversion.
|
2009-05-16 15:39:55 +08:00
|
|
|
|
// FIXME: Binding to a subobject of the lvalue is going to require more
|
|
|
|
|
// AST annotation than this.
|
Implement support for operator overloading using candidate operator
functions for built-in operators, e.g., the builtin
bool operator==(int const*, int const*)
can be used for the expression "x1 == x2" given:
struct X {
operator int const*();
} x1, x2;
The scheme for handling these built-in operators is relatively simple:
for each candidate required by the standard, create a special kind of
candidate function for the built-in. If overload resolution picks the
built-in operator, we perform the appropriate conversions on the
arguments and then let the normal built-in operator take care of it.
There may be some optimization opportunity left: if we can reduce the
number of built-in operator overloads we generate, overload resolution
for these cases will go faster. However, one must be careful when
doing this: GCC generates too few operator overloads in our little
test program, and fails to compile it because none of the overloads it
generates match.
Note that we only support operator overload for non-member binary
operators at the moment. The other operators will follow.
As part of this change, ImplicitCastExpr can now be an lvalue.
llvm-svn: 59148
2008-11-13 01:17:38 +08:00
|
|
|
|
ImpCastExprToType(Init, T1, /*isLvalue=*/true);
|
2008-11-11 00:14:15 +08:00
|
|
|
|
}
|
|
|
|
|
break;
|
2008-10-29 08:13:59 +08:00
|
|
|
|
|
2008-11-11 00:14:15 +08:00
|
|
|
|
case OR_Ambiguous:
|
|
|
|
|
assert(false && "Ambiguous reference binding conversions not implemented.");
|
|
|
|
|
return true;
|
|
|
|
|
|
|
|
|
|
case OR_No_Viable_Function:
|
2009-02-19 05:56:37 +08:00
|
|
|
|
case OR_Deleted:
|
|
|
|
|
// There was no suitable conversion, or we found a deleted
|
|
|
|
|
// conversion; continue with other checks.
|
2008-11-11 00:14:15 +08:00
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2008-10-29 08:13:59 +08:00
|
|
|
|
if (BindsDirectly) {
|
|
|
|
|
// C++ [dcl.init.ref]p4:
|
|
|
|
|
// [...] In all cases where the reference-related or
|
|
|
|
|
// reference-compatible relationship of two types is used to
|
|
|
|
|
// establish the validity of a reference binding, and T1 is a
|
|
|
|
|
// base class of T2, a program that necessitates such a binding
|
|
|
|
|
// is ill-formed if T1 is an inaccessible (clause 11) or
|
|
|
|
|
// ambiguous (10.2) base class of T2.
|
|
|
|
|
//
|
|
|
|
|
// Note that we only check this condition when we're allowed to
|
|
|
|
|
// complain about errors, because we should not be checking for
|
|
|
|
|
// ambiguity (or inaccessibility) unless the reference binding
|
|
|
|
|
// actually happens.
|
2008-10-29 10:00:59 +08:00
|
|
|
|
if (DerivedToBase)
|
|
|
|
|
return CheckDerivedToBaseConversion(T2, T1,
|
|
|
|
|
Init->getSourceRange().getBegin(),
|
|
|
|
|
Init->getSourceRange());
|
|
|
|
|
else
|
|
|
|
|
return false;
|
2008-10-29 08:13:59 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// -- Otherwise, the reference shall be to a non-volatile const
|
2009-03-29 23:27:50 +08:00
|
|
|
|
// type (i.e., cv1 shall be const), or the reference shall be an
|
|
|
|
|
// rvalue reference and the initializer expression shall be an rvalue.
|
2009-03-17 07:22:08 +08:00
|
|
|
|
if (!isRValRef && T1.getCVRQualifiers() != QualType::Const) {
|
2008-10-29 10:00:59 +08:00
|
|
|
|
if (!ICS)
|
2008-10-29 08:13:59 +08:00
|
|
|
|
Diag(Init->getSourceRange().getBegin(),
|
2008-11-19 06:52:51 +08:00
|
|
|
|
diag::err_not_reference_to_const_init)
|
2008-11-24 14:25:27 +08:00
|
|
|
|
<< T1 << (InitLvalue != Expr::LV_Valid? "temporary" : "value")
|
|
|
|
|
<< T2 << Init->getSourceRange();
|
2008-10-29 08:13:59 +08:00
|
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// -- If the initializer expression is an rvalue, with T2 a
|
|
|
|
|
// class type, and “cv1 T1” is reference-compatible with
|
|
|
|
|
// “cv2 T2,” the reference is bound in one of the
|
|
|
|
|
// following ways (the choice is implementation-defined):
|
|
|
|
|
//
|
|
|
|
|
// -- The reference is bound to the object represented by
|
|
|
|
|
// the rvalue (see 3.10) or to a sub-object within that
|
|
|
|
|
// object.
|
|
|
|
|
//
|
|
|
|
|
// -- A temporary of type “cv1 T2” [sic] is created, and
|
|
|
|
|
// a constructor is called to copy the entire rvalue
|
|
|
|
|
// object into the temporary. The reference is bound to
|
|
|
|
|
// the temporary or to a sub-object within the
|
|
|
|
|
// temporary.
|
|
|
|
|
//
|
|
|
|
|
// The constructor that would be used to make the copy
|
|
|
|
|
// shall be callable whether or not the copy is actually
|
|
|
|
|
// done.
|
|
|
|
|
//
|
2009-03-29 23:27:50 +08:00
|
|
|
|
// Note that C++0x [dcl.init.ref]p5 takes away this implementation
|
2008-10-29 08:13:59 +08:00
|
|
|
|
// freedom, so we will always take the first option and never build
|
|
|
|
|
// a temporary in this case. FIXME: We will, however, have to check
|
|
|
|
|
// for the presence of a copy constructor in C++98/03 mode.
|
|
|
|
|
if (InitLvalue != Expr::LV_Valid && T2->isRecordType() &&
|
2008-10-29 10:00:59 +08:00
|
|
|
|
RefRelationship >= Ref_Compatible_With_Added_Qualification) {
|
|
|
|
|
if (ICS) {
|
|
|
|
|
ICS->ConversionKind = ImplicitConversionSequence::StandardConversion;
|
|
|
|
|
ICS->Standard.First = ICK_Identity;
|
|
|
|
|
ICS->Standard.Second = DerivedToBase? ICK_Derived_To_Base : ICK_Identity;
|
|
|
|
|
ICS->Standard.Third = ICK_Identity;
|
|
|
|
|
ICS->Standard.FromTypePtr = T2.getAsOpaquePtr();
|
|
|
|
|
ICS->Standard.ToTypePtr = T1.getAsOpaquePtr();
|
2008-10-29 22:50:44 +08:00
|
|
|
|
ICS->Standard.ReferenceBinding = true;
|
2009-03-29 23:27:50 +08:00
|
|
|
|
ICS->Standard.DirectBinding = false;
|
|
|
|
|
ICS->Standard.RRefBinding = isRValRef;
|
2009-04-18 00:30:52 +08:00
|
|
|
|
ICS->Standard.CopyConstructor = 0;
|
2008-10-29 10:00:59 +08:00
|
|
|
|
} else {
|
2009-05-16 15:39:55 +08:00
|
|
|
|
// FIXME: Binding to a subobject of the rvalue is going to require more
|
|
|
|
|
// AST annotation than this.
|
2009-05-19 08:38:24 +08:00
|
|
|
|
ImpCastExprToType(Init, T1, /*isLvalue=*/false);
|
2008-10-29 08:13:59 +08:00
|
|
|
|
}
|
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// -- Otherwise, a temporary of type “cv1 T1” is created and
|
|
|
|
|
// initialized from the initializer expression using the
|
|
|
|
|
// rules for a non-reference copy initialization (8.5). The
|
|
|
|
|
// reference is then bound to the temporary. If T1 is
|
|
|
|
|
// reference-related to T2, cv1 must be the same
|
|
|
|
|
// cv-qualification as, or greater cv-qualification than,
|
|
|
|
|
// cv2; otherwise, the program is ill-formed.
|
|
|
|
|
if (RefRelationship == Ref_Related) {
|
|
|
|
|
// If cv1 == cv2 or cv1 is a greater cv-qualified than cv2, then
|
|
|
|
|
// we would be reference-compatible or reference-compatible with
|
|
|
|
|
// added qualification. But that wasn't the case, so the reference
|
|
|
|
|
// initialization fails.
|
2008-10-29 10:00:59 +08:00
|
|
|
|
if (!ICS)
|
2008-10-29 08:13:59 +08:00
|
|
|
|
Diag(Init->getSourceRange().getBegin(),
|
2008-11-19 06:52:51 +08:00
|
|
|
|
diag::err_reference_init_drops_quals)
|
2008-11-24 14:25:27 +08:00
|
|
|
|
<< T1 << (InitLvalue != Expr::LV_Valid? "temporary" : "value")
|
|
|
|
|
<< T2 << Init->getSourceRange();
|
2008-10-29 08:13:59 +08:00
|
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
|
2009-01-31 07:27:23 +08:00
|
|
|
|
// If at least one of the types is a class type, the types are not
|
|
|
|
|
// related, and we aren't allowed any user conversions, the
|
|
|
|
|
// reference binding fails. This case is important for breaking
|
|
|
|
|
// recursion, since TryImplicitConversion below will attempt to
|
|
|
|
|
// create a temporary through the use of a copy constructor.
|
|
|
|
|
if (SuppressUserConversions && RefRelationship == Ref_Incompatible &&
|
|
|
|
|
(T1->isRecordType() || T2->isRecordType())) {
|
|
|
|
|
if (!ICS)
|
|
|
|
|
Diag(Init->getSourceRange().getBegin(),
|
|
|
|
|
diag::err_typecheck_convert_incompatible)
|
|
|
|
|
<< DeclType << Init->getType() << "initializing" << Init->getSourceRange();
|
|
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
|
2008-10-29 08:13:59 +08:00
|
|
|
|
// Actually try to convert the initializer to T1.
|
2008-10-29 10:00:59 +08:00
|
|
|
|
if (ICS) {
|
2009-03-29 23:27:50 +08:00
|
|
|
|
// C++ [over.ics.ref]p2:
|
|
|
|
|
//
|
|
|
|
|
// When a parameter of reference type is not bound directly to
|
|
|
|
|
// an argument expression, the conversion sequence is the one
|
|
|
|
|
// required to convert the argument expression to the
|
|
|
|
|
// underlying type of the reference according to
|
|
|
|
|
// 13.3.3.1. Conceptually, this conversion sequence corresponds
|
|
|
|
|
// to copy-initializing a temporary of the underlying type with
|
|
|
|
|
// the argument expression. Any difference in top-level
|
|
|
|
|
// cv-qualification is subsumed by the initialization itself
|
|
|
|
|
// and does not constitute a conversion.
|
2008-11-04 03:09:14 +08:00
|
|
|
|
*ICS = TryImplicitConversion(Init, T1, SuppressUserConversions);
|
2009-03-29 23:27:50 +08:00
|
|
|
|
// Of course, that's still a reference binding.
|
|
|
|
|
if (ICS->ConversionKind == ImplicitConversionSequence::StandardConversion) {
|
|
|
|
|
ICS->Standard.ReferenceBinding = true;
|
|
|
|
|
ICS->Standard.RRefBinding = isRValRef;
|
|
|
|
|
} else if(ICS->ConversionKind ==
|
|
|
|
|
ImplicitConversionSequence::UserDefinedConversion) {
|
|
|
|
|
ICS->UserDefined.After.ReferenceBinding = true;
|
|
|
|
|
ICS->UserDefined.After.RRefBinding = isRValRef;
|
|
|
|
|
}
|
2008-10-29 10:00:59 +08:00
|
|
|
|
return ICS->ConversionKind == ImplicitConversionSequence::BadConversion;
|
|
|
|
|
} else {
|
2008-12-20 01:40:08 +08:00
|
|
|
|
return PerformImplicitConversion(Init, T1, "initializing");
|
2008-10-29 10:00:59 +08:00
|
|
|
|
}
|
2008-10-29 08:13:59 +08:00
|
|
|
|
}
|
2008-11-07 06:13:31 +08:00
|
|
|
|
|
|
|
|
|
/// CheckOverloadedOperatorDeclaration - Check whether the declaration
|
|
|
|
|
/// of this overloaded operator is well-formed. If so, returns false;
|
|
|
|
|
/// otherwise, emits appropriate diagnostics and returns true.
|
|
|
|
|
bool Sema::CheckOverloadedOperatorDeclaration(FunctionDecl *FnDecl) {
|
2008-11-18 00:14:12 +08:00
|
|
|
|
assert(FnDecl && FnDecl->isOverloadedOperator() &&
|
2008-11-07 06:13:31 +08:00
|
|
|
|
"Expected an overloaded operator declaration");
|
|
|
|
|
|
|
|
|
|
OverloadedOperatorKind Op = FnDecl->getOverloadedOperator();
|
|
|
|
|
|
|
|
|
|
// C++ [over.oper]p5:
|
|
|
|
|
// The allocation and deallocation functions, operator new,
|
|
|
|
|
// operator new[], operator delete and operator delete[], are
|
|
|
|
|
// described completely in 3.7.3. The attributes and restrictions
|
|
|
|
|
// found in the rest of this subclause do not apply to them unless
|
|
|
|
|
// explicitly stated in 3.7.3.
|
2009-05-16 15:39:55 +08:00
|
|
|
|
// FIXME: Write a separate routine for checking this. For now, just allow it.
|
2008-11-07 06:13:31 +08:00
|
|
|
|
if (Op == OO_New || Op == OO_Array_New ||
|
|
|
|
|
Op == OO_Delete || Op == OO_Array_Delete)
|
|
|
|
|
return false;
|
|
|
|
|
|
|
|
|
|
// C++ [over.oper]p6:
|
|
|
|
|
// An operator function shall either be a non-static member
|
|
|
|
|
// function or be a non-member function and have at least one
|
|
|
|
|
// parameter whose type is a class, a reference to a class, an
|
|
|
|
|
// enumeration, or a reference to an enumeration.
|
2008-11-18 00:14:12 +08:00
|
|
|
|
if (CXXMethodDecl *MethodDecl = dyn_cast<CXXMethodDecl>(FnDecl)) {
|
|
|
|
|
if (MethodDecl->isStatic())
|
|
|
|
|
return Diag(FnDecl->getLocation(),
|
2008-11-24 13:29:24 +08:00
|
|
|
|
diag::err_operator_overload_static) << FnDecl->getDeclName();
|
2008-11-07 06:13:31 +08:00
|
|
|
|
} else {
|
|
|
|
|
bool ClassOrEnumParam = false;
|
2008-11-18 00:14:12 +08:00
|
|
|
|
for (FunctionDecl::param_iterator Param = FnDecl->param_begin(),
|
|
|
|
|
ParamEnd = FnDecl->param_end();
|
|
|
|
|
Param != ParamEnd; ++Param) {
|
|
|
|
|
QualType ParamType = (*Param)->getType().getNonReferenceType();
|
2008-11-07 06:13:31 +08:00
|
|
|
|
if (ParamType->isRecordType() || ParamType->isEnumeralType()) {
|
|
|
|
|
ClassOrEnumParam = true;
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2008-11-18 00:14:12 +08:00
|
|
|
|
if (!ClassOrEnumParam)
|
|
|
|
|
return Diag(FnDecl->getLocation(),
|
2008-11-20 14:38:18 +08:00
|
|
|
|
diag::err_operator_overload_needs_class_or_enum)
|
2008-11-24 13:29:24 +08:00
|
|
|
|
<< FnDecl->getDeclName();
|
2008-11-07 06:13:31 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// C++ [over.oper]p8:
|
|
|
|
|
// An operator function cannot have default arguments (8.3.6),
|
|
|
|
|
// except where explicitly stated below.
|
|
|
|
|
//
|
|
|
|
|
// Only the function-call operator allows default arguments
|
|
|
|
|
// (C++ [over.call]p1).
|
|
|
|
|
if (Op != OO_Call) {
|
|
|
|
|
for (FunctionDecl::param_iterator Param = FnDecl->param_begin();
|
|
|
|
|
Param != FnDecl->param_end(); ++Param) {
|
2008-12-24 08:01:03 +08:00
|
|
|
|
if ((*Param)->hasUnparsedDefaultArg())
|
|
|
|
|
return Diag((*Param)->getLocation(),
|
|
|
|
|
diag::err_operator_overload_default_arg)
|
|
|
|
|
<< FnDecl->getDeclName();
|
|
|
|
|
else if (Expr *DefArg = (*Param)->getDefaultArg())
|
2008-11-18 00:14:12 +08:00
|
|
|
|
return Diag((*Param)->getLocation(),
|
2008-11-20 14:06:08 +08:00
|
|
|
|
diag::err_operator_overload_default_arg)
|
2008-11-24 13:29:24 +08:00
|
|
|
|
<< FnDecl->getDeclName() << DefArg->getSourceRange();
|
2008-11-07 06:13:31 +08:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2008-11-10 21:38:07 +08:00
|
|
|
|
static const bool OperatorUses[NUM_OVERLOADED_OPERATORS][3] = {
|
|
|
|
|
{ false, false, false }
|
|
|
|
|
#define OVERLOADED_OPERATOR(Name,Spelling,Token,Unary,Binary,MemberOnly) \
|
|
|
|
|
, { Unary, Binary, MemberOnly }
|
|
|
|
|
#include "clang/Basic/OperatorKinds.def"
|
|
|
|
|
};
|
2008-11-07 06:13:31 +08:00
|
|
|
|
|
2008-11-10 21:38:07 +08:00
|
|
|
|
bool CanBeUnaryOperator = OperatorUses[Op][0];
|
|
|
|
|
bool CanBeBinaryOperator = OperatorUses[Op][1];
|
|
|
|
|
bool MustBeMemberOperator = OperatorUses[Op][2];
|
2008-11-07 06:13:31 +08:00
|
|
|
|
|
|
|
|
|
// C++ [over.oper]p8:
|
|
|
|
|
// [...] Operator functions cannot have more or fewer parameters
|
|
|
|
|
// than the number required for the corresponding operator, as
|
|
|
|
|
// described in the rest of this subclause.
|
2008-11-18 00:14:12 +08:00
|
|
|
|
unsigned NumParams = FnDecl->getNumParams()
|
|
|
|
|
+ (isa<CXXMethodDecl>(FnDecl)? 1 : 0);
|
2008-11-07 06:13:31 +08:00
|
|
|
|
if (Op != OO_Call &&
|
|
|
|
|
((NumParams == 1 && !CanBeUnaryOperator) ||
|
|
|
|
|
(NumParams == 2 && !CanBeBinaryOperator) ||
|
|
|
|
|
(NumParams < 1) || (NumParams > 2))) {
|
|
|
|
|
// We have the wrong number of parameters.
|
2008-11-21 15:57:12 +08:00
|
|
|
|
unsigned ErrorKind;
|
2008-11-10 21:38:07 +08:00
|
|
|
|
if (CanBeUnaryOperator && CanBeBinaryOperator) {
|
2008-11-21 15:57:12 +08:00
|
|
|
|
ErrorKind = 2; // 2 -> unary or binary.
|
2008-11-10 21:38:07 +08:00
|
|
|
|
} else if (CanBeUnaryOperator) {
|
2008-11-21 15:57:12 +08:00
|
|
|
|
ErrorKind = 0; // 0 -> unary
|
2008-11-10 21:38:07 +08:00
|
|
|
|
} else {
|
2008-11-21 15:50:02 +08:00
|
|
|
|
assert(CanBeBinaryOperator &&
|
|
|
|
|
"All non-call overloaded operators are unary or binary!");
|
2008-11-21 15:57:12 +08:00
|
|
|
|
ErrorKind = 1; // 1 -> binary
|
2008-11-10 21:38:07 +08:00
|
|
|
|
}
|
2008-11-07 06:13:31 +08:00
|
|
|
|
|
2008-11-21 15:57:12 +08:00
|
|
|
|
return Diag(FnDecl->getLocation(), diag::err_operator_overload_must_be)
|
2008-11-24 13:29:24 +08:00
|
|
|
|
<< FnDecl->getDeclName() << NumParams << ErrorKind;
|
2008-11-07 06:13:31 +08:00
|
|
|
|
}
|
2009-01-06 04:52:13 +08:00
|
|
|
|
|
2008-11-18 00:14:12 +08:00
|
|
|
|
// Overloaded operators other than operator() cannot be variadic.
|
|
|
|
|
if (Op != OO_Call &&
|
2009-02-27 07:50:07 +08:00
|
|
|
|
FnDecl->getType()->getAsFunctionProtoType()->isVariadic()) {
|
2008-11-20 14:38:18 +08:00
|
|
|
|
return Diag(FnDecl->getLocation(), diag::err_operator_overload_variadic)
|
2008-11-24 13:29:24 +08:00
|
|
|
|
<< FnDecl->getDeclName();
|
2008-11-07 06:13:31 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Some operators must be non-static member functions.
|
2008-11-18 00:14:12 +08:00
|
|
|
|
if (MustBeMemberOperator && !isa<CXXMethodDecl>(FnDecl)) {
|
|
|
|
|
return Diag(FnDecl->getLocation(),
|
2008-11-20 14:38:18 +08:00
|
|
|
|
diag::err_operator_overload_must_be_member)
|
2008-11-24 13:29:24 +08:00
|
|
|
|
<< FnDecl->getDeclName();
|
2008-11-07 06:13:31 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// C++ [over.inc]p1:
|
|
|
|
|
// The user-defined function called operator++ implements the
|
|
|
|
|
// prefix and postfix ++ operator. If this function is a member
|
|
|
|
|
// function with no parameters, or a non-member function with one
|
|
|
|
|
// parameter of class or enumeration type, it defines the prefix
|
|
|
|
|
// increment operator ++ for objects of that type. If the function
|
|
|
|
|
// is a member function with one parameter (which shall be of type
|
|
|
|
|
// int) or a non-member function with two parameters (the second
|
|
|
|
|
// of which shall be of type int), it defines the postfix
|
|
|
|
|
// increment operator ++ for objects of that type.
|
|
|
|
|
if ((Op == OO_PlusPlus || Op == OO_MinusMinus) && NumParams == 2) {
|
|
|
|
|
ParmVarDecl *LastParam = FnDecl->getParamDecl(FnDecl->getNumParams() - 1);
|
|
|
|
|
bool ParamIsInt = false;
|
|
|
|
|
if (const BuiltinType *BT = LastParam->getType()->getAsBuiltinType())
|
|
|
|
|
ParamIsInt = BT->getKind() == BuiltinType::Int;
|
|
|
|
|
|
2008-11-21 15:50:02 +08:00
|
|
|
|
if (!ParamIsInt)
|
|
|
|
|
return Diag(LastParam->getLocation(),
|
|
|
|
|
diag::err_operator_overload_post_incdec_must_be_int)
|
2008-11-24 14:25:27 +08:00
|
|
|
|
<< LastParam->getType() << (Op == OO_MinusMinus);
|
2008-11-07 06:13:31 +08:00
|
|
|
|
}
|
|
|
|
|
|
2009-01-06 04:52:13 +08:00
|
|
|
|
// Notify the class if it got an assignment operator.
|
|
|
|
|
if (Op == OO_Equal) {
|
|
|
|
|
// Would have returned earlier otherwise.
|
|
|
|
|
assert(isa<CXXMethodDecl>(FnDecl) &&
|
|
|
|
|
"Overloaded = not member, but not filtered.");
|
|
|
|
|
CXXMethodDecl *Method = cast<CXXMethodDecl>(FnDecl);
|
|
|
|
|
Method->getParent()->addedAssignmentOperator(Context, Method);
|
|
|
|
|
}
|
|
|
|
|
|
2008-11-18 00:14:12 +08:00
|
|
|
|
return false;
|
2008-11-07 06:13:31 +08:00
|
|
|
|
}
|
2008-12-17 15:09:26 +08:00
|
|
|
|
|
2009-01-06 03:45:36 +08:00
|
|
|
|
/// ActOnStartLinkageSpecification - Parsed the beginning of a C++
|
|
|
|
|
/// linkage specification, including the language and (if present)
|
|
|
|
|
/// the '{'. ExternLoc is the location of the 'extern', LangLoc is
|
|
|
|
|
/// the location of the language string literal, which is provided
|
|
|
|
|
/// by Lang/StrSize. LBraceLoc, if valid, provides the location of
|
|
|
|
|
/// the '{' brace. Otherwise, this linkage specification does not
|
|
|
|
|
/// have any braces.
|
2009-03-29 03:18:32 +08:00
|
|
|
|
Sema::DeclPtrTy Sema::ActOnStartLinkageSpecification(Scope *S,
|
|
|
|
|
SourceLocation ExternLoc,
|
|
|
|
|
SourceLocation LangLoc,
|
|
|
|
|
const char *Lang,
|
|
|
|
|
unsigned StrSize,
|
|
|
|
|
SourceLocation LBraceLoc) {
|
2008-12-17 15:13:27 +08:00
|
|
|
|
LinkageSpecDecl::LanguageIDs Language;
|
|
|
|
|
if (strncmp(Lang, "\"C\"", StrSize) == 0)
|
|
|
|
|
Language = LinkageSpecDecl::lang_c;
|
|
|
|
|
else if (strncmp(Lang, "\"C++\"", StrSize) == 0)
|
|
|
|
|
Language = LinkageSpecDecl::lang_cxx;
|
|
|
|
|
else {
|
2009-01-06 03:45:36 +08:00
|
|
|
|
Diag(LangLoc, diag::err_bad_language);
|
2009-03-29 03:18:32 +08:00
|
|
|
|
return DeclPtrTy();
|
2008-12-17 15:13:27 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// FIXME: Add all the various semantics of linkage specifications
|
|
|
|
|
|
2009-01-06 03:45:36 +08:00
|
|
|
|
LinkageSpecDecl *D = LinkageSpecDecl::Create(Context, CurContext,
|
|
|
|
|
LangLoc, Language,
|
|
|
|
|
LBraceLoc.isValid());
|
2009-04-10 05:40:53 +08:00
|
|
|
|
CurContext->addDecl(Context, D);
|
2009-01-06 03:45:36 +08:00
|
|
|
|
PushDeclContext(S, D);
|
2009-03-29 03:18:32 +08:00
|
|
|
|
return DeclPtrTy::make(D);
|
2008-12-17 15:13:27 +08:00
|
|
|
|
}
|
|
|
|
|
|
2009-01-06 03:45:36 +08:00
|
|
|
|
/// ActOnFinishLinkageSpecification - Completely the definition of
|
|
|
|
|
/// the C++ linkage specification LinkageSpec. If RBraceLoc is
|
|
|
|
|
/// valid, it's the position of the closing '}' brace in a linkage
|
|
|
|
|
/// specification that uses braces.
|
2009-03-29 03:18:32 +08:00
|
|
|
|
Sema::DeclPtrTy Sema::ActOnFinishLinkageSpecification(Scope *S,
|
|
|
|
|
DeclPtrTy LinkageSpec,
|
|
|
|
|
SourceLocation RBraceLoc) {
|
2009-01-06 03:45:36 +08:00
|
|
|
|
if (LinkageSpec)
|
|
|
|
|
PopDeclContext();
|
|
|
|
|
return LinkageSpec;
|
2008-12-17 15:09:26 +08:00
|
|
|
|
}
|
|
|
|
|
|
2009-05-19 04:51:54 +08:00
|
|
|
|
/// \brief Perform semantic analysis for the variable declaration that
|
|
|
|
|
/// occurs within a C++ catch clause, returning the newly-created
|
|
|
|
|
/// variable.
|
|
|
|
|
VarDecl *Sema::BuildExceptionDeclaration(Scope *S, QualType ExDeclType,
|
|
|
|
|
IdentifierInfo *Name,
|
|
|
|
|
SourceLocation Loc,
|
|
|
|
|
SourceRange Range) {
|
|
|
|
|
bool Invalid = false;
|
2008-12-23 03:15:10 +08:00
|
|
|
|
|
|
|
|
|
// Arrays and functions decay.
|
|
|
|
|
if (ExDeclType->isArrayType())
|
|
|
|
|
ExDeclType = Context.getArrayDecayedType(ExDeclType);
|
|
|
|
|
else if (ExDeclType->isFunctionType())
|
|
|
|
|
ExDeclType = Context.getPointerType(ExDeclType);
|
|
|
|
|
|
|
|
|
|
// C++ 15.3p1: The exception-declaration shall not denote an incomplete type.
|
|
|
|
|
// The exception-declaration shall not denote a pointer or reference to an
|
|
|
|
|
// incomplete type, other than [cv] void*.
|
2009-03-23 07:49:27 +08:00
|
|
|
|
// N2844 forbids rvalue references.
|
2009-05-19 05:08:14 +08:00
|
|
|
|
if(!ExDeclType->isDependentType() && ExDeclType->isRValueReferenceType()) {
|
2009-05-19 04:51:54 +08:00
|
|
|
|
Diag(Loc, diag::err_catch_rvalue_ref) << Range;
|
2009-03-23 07:49:27 +08:00
|
|
|
|
Invalid = true;
|
|
|
|
|
}
|
2009-05-19 04:51:54 +08:00
|
|
|
|
|
2008-12-23 03:15:10 +08:00
|
|
|
|
QualType BaseType = ExDeclType;
|
|
|
|
|
int Mode = 0; // 0 for direct type, 1 for pointer, 2 for reference
|
2009-01-20 03:26:10 +08:00
|
|
|
|
unsigned DK = diag::err_catch_incomplete;
|
2008-12-23 03:15:10 +08:00
|
|
|
|
if (const PointerType *Ptr = BaseType->getAsPointerType()) {
|
|
|
|
|
BaseType = Ptr->getPointeeType();
|
|
|
|
|
Mode = 1;
|
2009-01-20 03:26:10 +08:00
|
|
|
|
DK = diag::err_catch_incomplete_ptr;
|
2008-12-23 03:15:10 +08:00
|
|
|
|
} else if(const ReferenceType *Ref = BaseType->getAsReferenceType()) {
|
2009-03-23 07:49:27 +08:00
|
|
|
|
// For the purpose of error recovery, we treat rvalue refs like lvalue refs.
|
2008-12-23 03:15:10 +08:00
|
|
|
|
BaseType = Ref->getPointeeType();
|
|
|
|
|
Mode = 2;
|
2009-01-20 03:26:10 +08:00
|
|
|
|
DK = diag::err_catch_incomplete_ref;
|
2008-12-23 03:15:10 +08:00
|
|
|
|
}
|
2009-03-23 07:49:27 +08:00
|
|
|
|
if (!Invalid && (Mode == 0 || !BaseType->isVoidType()) &&
|
2009-05-19 04:51:54 +08:00
|
|
|
|
!BaseType->isDependentType() && RequireCompleteType(Loc, BaseType, DK))
|
2008-12-23 03:15:10 +08:00
|
|
|
|
Invalid = true;
|
|
|
|
|
|
2009-05-19 04:51:54 +08:00
|
|
|
|
if (!Invalid && !ExDeclType->isDependentType() &&
|
|
|
|
|
RequireNonAbstractType(Loc, ExDeclType,
|
|
|
|
|
diag::err_abstract_type_in_decl,
|
|
|
|
|
AbstractVariableType))
|
2009-04-28 05:03:30 +08:00
|
|
|
|
Invalid = true;
|
|
|
|
|
|
2009-05-19 04:51:54 +08:00
|
|
|
|
// FIXME: Need to test for ability to copy-construct and destroy the
|
|
|
|
|
// exception variable.
|
|
|
|
|
|
2008-12-23 05:35:02 +08:00
|
|
|
|
// FIXME: Need to check for abstract classes.
|
|
|
|
|
|
2009-05-19 04:51:54 +08:00
|
|
|
|
VarDecl *ExDecl = VarDecl::Create(Context, CurContext, Loc,
|
|
|
|
|
Name, ExDeclType, VarDecl::None,
|
|
|
|
|
Range.getBegin());
|
|
|
|
|
|
|
|
|
|
if (Invalid)
|
|
|
|
|
ExDecl->setInvalidDecl();
|
|
|
|
|
|
|
|
|
|
return ExDecl;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/// ActOnExceptionDeclarator - Parsed the exception-declarator in a C++ catch
|
|
|
|
|
/// handler.
|
|
|
|
|
Sema::DeclPtrTy Sema::ActOnExceptionDeclarator(Scope *S, Declarator &D) {
|
|
|
|
|
QualType ExDeclType = GetTypeForDeclarator(D, S);
|
|
|
|
|
|
|
|
|
|
bool Invalid = D.isInvalidType();
|
2008-12-23 03:15:10 +08:00
|
|
|
|
IdentifierInfo *II = D.getIdentifier();
|
2009-02-05 01:27:36 +08:00
|
|
|
|
if (NamedDecl *PrevDecl = LookupName(S, II, LookupOrdinaryName)) {
|
2008-12-23 03:15:10 +08:00
|
|
|
|
// The scope should be freshly made just for us. There is just no way
|
|
|
|
|
// it contains any previous declaration.
|
2009-03-29 03:18:32 +08:00
|
|
|
|
assert(!S->isDeclScope(DeclPtrTy::make(PrevDecl)));
|
2008-12-23 03:15:10 +08:00
|
|
|
|
if (PrevDecl->isTemplateParameter()) {
|
|
|
|
|
// Maybe we will complain about the shadowed template parameter.
|
|
|
|
|
DiagnoseTemplateParameterShadow(D.getIdentifierLoc(), PrevDecl);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2009-04-25 16:06:05 +08:00
|
|
|
|
if (D.getCXXScopeSpec().isSet() && !Invalid) {
|
2008-12-23 03:15:10 +08:00
|
|
|
|
Diag(D.getIdentifierLoc(), diag::err_qualified_catch_declarator)
|
|
|
|
|
<< D.getCXXScopeSpec().getRange();
|
2009-04-25 16:06:05 +08:00
|
|
|
|
Invalid = true;
|
2008-12-23 03:15:10 +08:00
|
|
|
|
}
|
|
|
|
|
|
2009-05-19 04:51:54 +08:00
|
|
|
|
VarDecl *ExDecl = BuildExceptionDeclaration(S, ExDeclType,
|
|
|
|
|
D.getIdentifier(),
|
|
|
|
|
D.getIdentifierLoc(),
|
|
|
|
|
D.getDeclSpec().getSourceRange());
|
|
|
|
|
|
2009-04-25 16:06:05 +08:00
|
|
|
|
if (Invalid)
|
|
|
|
|
ExDecl->setInvalidDecl();
|
|
|
|
|
|
2008-12-23 03:15:10 +08:00
|
|
|
|
// Add the exception declaration into this scope.
|
|
|
|
|
if (II)
|
2009-05-19 04:51:54 +08:00
|
|
|
|
PushOnScopeChains(ExDecl, S);
|
|
|
|
|
else
|
|
|
|
|
CurContext->addDecl(Context, ExDecl);
|
2008-12-23 03:15:10 +08:00
|
|
|
|
|
|
|
|
|
ProcessDeclAttributes(ExDecl, D);
|
2009-03-29 03:18:32 +08:00
|
|
|
|
return DeclPtrTy::make(ExDecl);
|
2008-12-23 03:15:10 +08:00
|
|
|
|
}
|
2009-03-14 08:25:26 +08:00
|
|
|
|
|
2009-03-29 03:18:32 +08:00
|
|
|
|
Sema::DeclPtrTy Sema::ActOnStaticAssertDeclaration(SourceLocation AssertLoc,
|
|
|
|
|
ExprArg assertexpr,
|
|
|
|
|
ExprArg assertmessageexpr) {
|
2009-03-14 08:25:26 +08:00
|
|
|
|
Expr *AssertExpr = (Expr *)assertexpr.get();
|
|
|
|
|
StringLiteral *AssertMessage =
|
|
|
|
|
cast<StringLiteral>((Expr *)assertmessageexpr.get());
|
|
|
|
|
|
2009-03-14 08:33:21 +08:00
|
|
|
|
if (!AssertExpr->isTypeDependent() && !AssertExpr->isValueDependent()) {
|
|
|
|
|
llvm::APSInt Value(32);
|
|
|
|
|
if (!AssertExpr->isIntegerConstantExpr(Value, Context)) {
|
|
|
|
|
Diag(AssertLoc, diag::err_static_assert_expression_is_not_constant) <<
|
|
|
|
|
AssertExpr->getSourceRange();
|
2009-03-29 03:18:32 +08:00
|
|
|
|
return DeclPtrTy();
|
2009-03-14 08:33:21 +08:00
|
|
|
|
}
|
2009-03-14 08:25:26 +08:00
|
|
|
|
|
2009-03-14 08:33:21 +08:00
|
|
|
|
if (Value == 0) {
|
|
|
|
|
std::string str(AssertMessage->getStrData(),
|
|
|
|
|
AssertMessage->getByteLength());
|
2009-03-16 02:44:04 +08:00
|
|
|
|
Diag(AssertLoc, diag::err_static_assert_failed)
|
|
|
|
|
<< str << AssertExpr->getSourceRange();
|
2009-03-14 08:33:21 +08:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2009-03-16 01:35:16 +08:00
|
|
|
|
assertexpr.release();
|
|
|
|
|
assertmessageexpr.release();
|
2009-03-14 08:25:26 +08:00
|
|
|
|
Decl *Decl = StaticAssertDecl::Create(Context, CurContext, AssertLoc,
|
|
|
|
|
AssertExpr, AssertMessage);
|
|
|
|
|
|
2009-04-10 05:40:53 +08:00
|
|
|
|
CurContext->addDecl(Context, Decl);
|
2009-03-29 03:18:32 +08:00
|
|
|
|
return DeclPtrTy::make(Decl);
|
2009-03-14 08:25:26 +08:00
|
|
|
|
}
|
2009-03-25 06:27:57 +08:00
|
|
|
|
|
2009-05-12 06:55:49 +08:00
|
|
|
|
bool Sema::ActOnFriendDecl(Scope *S, SourceLocation FriendLoc, DeclPtrTy Dcl) {
|
|
|
|
|
if (!(S->getFlags() & Scope::ClassScope)) {
|
|
|
|
|
Diag(FriendLoc, diag::err_friend_decl_outside_class);
|
|
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
|
2009-03-29 03:18:32 +08:00
|
|
|
|
void Sema::SetDeclDeleted(DeclPtrTy dcl, SourceLocation DelLoc) {
|
|
|
|
|
Decl *Dcl = dcl.getAs<Decl>();
|
2009-03-25 06:27:57 +08:00
|
|
|
|
FunctionDecl *Fn = dyn_cast<FunctionDecl>(Dcl);
|
|
|
|
|
if (!Fn) {
|
|
|
|
|
Diag(DelLoc, diag::err_deleted_non_function);
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
if (const FunctionDecl *Prev = Fn->getPreviousDeclaration()) {
|
|
|
|
|
Diag(DelLoc, diag::err_deleted_decl_not_first);
|
|
|
|
|
Diag(Prev->getLocation(), diag::note_previous_declaration);
|
|
|
|
|
// If the declaration wasn't the first, we delete the function anyway for
|
|
|
|
|
// recovery.
|
|
|
|
|
}
|
|
|
|
|
Fn->setDeleted();
|
|
|
|
|
}
|
2009-04-28 05:33:24 +08:00
|
|
|
|
|
|
|
|
|
static void SearchForReturnInStmt(Sema &Self, Stmt *S) {
|
|
|
|
|
for (Stmt::child_iterator CI = S->child_begin(), E = S->child_end(); CI != E;
|
|
|
|
|
++CI) {
|
|
|
|
|
Stmt *SubStmt = *CI;
|
|
|
|
|
if (!SubStmt)
|
|
|
|
|
continue;
|
|
|
|
|
if (isa<ReturnStmt>(SubStmt))
|
|
|
|
|
Self.Diag(SubStmt->getSourceRange().getBegin(),
|
|
|
|
|
diag::err_return_in_constructor_handler);
|
|
|
|
|
if (!isa<Expr>(SubStmt))
|
|
|
|
|
SearchForReturnInStmt(Self, SubStmt);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void Sema::DiagnoseReturnInConstructorExceptionHandler(CXXTryStmt *TryBlock) {
|
|
|
|
|
for (unsigned I = 0, E = TryBlock->getNumHandlers(); I != E; ++I) {
|
|
|
|
|
CXXCatchStmt *Handler = TryBlock->getHandler(I);
|
|
|
|
|
SearchForReturnInStmt(*this, Handler);
|
|
|
|
|
}
|
|
|
|
|
}
|
2009-05-14 09:09:04 +08:00
|
|
|
|
|
|
|
|
|
bool Sema::CheckOverridingFunctionReturnType(const CXXMethodDecl *New,
|
|
|
|
|
const CXXMethodDecl *Old) {
|
|
|
|
|
QualType NewTy = New->getType()->getAsFunctionType()->getResultType();
|
|
|
|
|
QualType OldTy = Old->getType()->getAsFunctionType()->getResultType();
|
|
|
|
|
|
|
|
|
|
QualType CNewTy = Context.getCanonicalType(NewTy);
|
|
|
|
|
QualType COldTy = Context.getCanonicalType(OldTy);
|
|
|
|
|
|
|
|
|
|
if (CNewTy == COldTy &&
|
|
|
|
|
CNewTy.getCVRQualifiers() == COldTy.getCVRQualifiers())
|
|
|
|
|
return false;
|
|
|
|
|
|
2009-05-15 03:52:19 +08:00
|
|
|
|
// Check if the return types are covariant
|
|
|
|
|
QualType NewClassTy, OldClassTy;
|
|
|
|
|
|
|
|
|
|
/// Both types must be pointers or references to classes.
|
|
|
|
|
if (PointerType *NewPT = dyn_cast<PointerType>(NewTy)) {
|
|
|
|
|
if (PointerType *OldPT = dyn_cast<PointerType>(OldTy)) {
|
|
|
|
|
NewClassTy = NewPT->getPointeeType();
|
|
|
|
|
OldClassTy = OldPT->getPointeeType();
|
|
|
|
|
}
|
|
|
|
|
} else if (ReferenceType *NewRT = dyn_cast<ReferenceType>(NewTy)) {
|
|
|
|
|
if (ReferenceType *OldRT = dyn_cast<ReferenceType>(OldTy)) {
|
|
|
|
|
NewClassTy = NewRT->getPointeeType();
|
|
|
|
|
OldClassTy = OldRT->getPointeeType();
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// The return types aren't either both pointers or references to a class type.
|
|
|
|
|
if (NewClassTy.isNull()) {
|
|
|
|
|
Diag(New->getLocation(),
|
|
|
|
|
diag::err_different_return_type_for_overriding_virtual_function)
|
|
|
|
|
<< New->getDeclName() << NewTy << OldTy;
|
|
|
|
|
Diag(Old->getLocation(), diag::note_overridden_virtual_function);
|
|
|
|
|
|
|
|
|
|
return true;
|
|
|
|
|
}
|
2009-05-14 09:09:04 +08:00
|
|
|
|
|
2009-05-15 03:52:19 +08:00
|
|
|
|
if (NewClassTy.getUnqualifiedType() != OldClassTy.getUnqualifiedType()) {
|
|
|
|
|
// Check if the new class derives from the old class.
|
|
|
|
|
if (!IsDerivedFrom(NewClassTy, OldClassTy)) {
|
|
|
|
|
Diag(New->getLocation(),
|
|
|
|
|
diag::err_covariant_return_not_derived)
|
|
|
|
|
<< New->getDeclName() << NewTy << OldTy;
|
|
|
|
|
Diag(Old->getLocation(), diag::note_overridden_virtual_function);
|
|
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Check if we the conversion from derived to base is valid.
|
|
|
|
|
if (CheckDerivedToBaseConversion(NewClassTy, OldClassTy,
|
|
|
|
|
diag::err_covariant_return_inaccessible_base,
|
|
|
|
|
diag::err_covariant_return_ambiguous_derived_to_base_conv,
|
|
|
|
|
// FIXME: Should this point to the return type?
|
|
|
|
|
New->getLocation(), SourceRange(), New->getDeclName())) {
|
|
|
|
|
Diag(Old->getLocation(), diag::note_overridden_virtual_function);
|
|
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// The qualifiers of the return types must be the same.
|
|
|
|
|
if (CNewTy.getCVRQualifiers() != COldTy.getCVRQualifiers()) {
|
|
|
|
|
Diag(New->getLocation(),
|
|
|
|
|
diag::err_covariant_return_type_different_qualifications)
|
2009-05-14 09:09:04 +08:00
|
|
|
|
<< New->getDeclName() << NewTy << OldTy;
|
2009-05-15 03:52:19 +08:00
|
|
|
|
Diag(Old->getLocation(), diag::note_overridden_virtual_function);
|
|
|
|
|
return true;
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
// The new class type must have the same or less qualifiers as the old type.
|
|
|
|
|
if (NewClassTy.isMoreQualifiedThan(OldClassTy)) {
|
|
|
|
|
Diag(New->getLocation(),
|
|
|
|
|
diag::err_covariant_return_type_class_type_more_qualified)
|
|
|
|
|
<< New->getDeclName() << NewTy << OldTy;
|
|
|
|
|
Diag(Old->getLocation(), diag::note_overridden_virtual_function);
|
|
|
|
|
return true;
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
return false;
|
2009-05-14 09:09:04 +08:00
|
|
|
|
}
|