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"
|
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/Basic/Diagnostic.h"
|
|
|
|
|
#include "clang/Parse/DeclSpec.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
|
|
|
|
|
Sema::ActOnParamDefaultArgument(DeclTy *param, SourceLocation EqualLoc,
|
|
|
|
|
ExprTy *defarg) {
|
|
|
|
|
ParmVarDecl *Param = (ParmVarDecl *)param;
|
|
|
|
|
llvm::OwningPtr<Expr> DefaultArg((Expr *)defarg);
|
|
|
|
|
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-11-04 21:57:51 +08:00
|
|
|
|
bool DefaultInitFailed = PerformCopyInitialization(DefaultArgPtr, ParamType,
|
|
|
|
|
"in default argument");
|
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
|
|
|
|
|
2008-04-08 13:04:30 +08:00
|
|
|
|
// Okay: add the default argument to the parameter
|
|
|
|
|
Param->setDefaultArg(DefaultArg.take());
|
|
|
|
|
}
|
|
|
|
|
|
2008-12-17 05:30:33 +08:00
|
|
|
|
/// ActOnParamDefaultArgumentError - Parsing or semantic analysis of
|
|
|
|
|
/// the default argument for the parameter param failed.
|
|
|
|
|
void Sema::ActOnParamDefaultArgumentError(DeclTy *param) {
|
|
|
|
|
((ParmVarDecl*)param)->setInvalidDecl();
|
|
|
|
|
}
|
|
|
|
|
|
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.
|
|
|
|
|
for (unsigned i = 0; i < D.getNumTypeObjects(); ++i) {
|
|
|
|
|
DeclaratorChunk &chunk = D.getTypeObject(i);
|
|
|
|
|
if (chunk.Kind == DeclaratorChunk::Function) {
|
|
|
|
|
for (unsigned argIdx = 0; argIdx < chunk.Fun.NumArgs; ++argIdx) {
|
|
|
|
|
ParmVarDecl *Param = (ParmVarDecl *)chunk.Fun.ArgInfo[argIdx].Param;
|
|
|
|
|
if (Param->getDefaultArg()) {
|
2008-11-19 13:08:23 +08:00
|
|
|
|
Diag(Param->getLocation(), diag::err_param_default_argument_nonfunc)
|
|
|
|
|
<< Param->getDefaultArg()->getSourceRange();
|
2008-05-07 12:49:29 +08:00
|
|
|
|
Param->setDefaultArg(0);
|
2008-12-17 05:30:33 +08:00
|
|
|
|
} else if (CachedTokens *Toks
|
|
|
|
|
= chunk.Fun.ArgInfo[argIdx].DefaultArgTokens) {
|
|
|
|
|
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-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
|
|
|
|
|
// type. Subroutine of MergeFunctionDecl.
|
|
|
|
|
FunctionDecl *
|
|
|
|
|
Sema::MergeCXXFunctionDecl(FunctionDecl *New, FunctionDecl *Old) {
|
|
|
|
|
// 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);
|
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());
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return New;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/// 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);
|
|
|
|
|
if (Param->getDefaultArg()) {
|
|
|
|
|
delete Param->getDefaultArg();
|
|
|
|
|
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;
|
|
|
|
|
if (SS) {
|
|
|
|
|
DeclContext *DC = static_cast<DeclContext*>(SS->getScopeRep());
|
|
|
|
|
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;
|
|
|
|
|
}
|
|
|
|
|
|
2008-04-14 05:30:24 +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.
|
2008-10-23 01:49:05 +08:00
|
|
|
|
Sema::BaseResult
|
|
|
|
|
Sema::ActOnBaseSpecifier(DeclTy *classdecl, SourceRange SpecifierRange,
|
|
|
|
|
bool Virtual, AccessSpecifier Access,
|
|
|
|
|
TypeTy *basetype, SourceLocation BaseLoc) {
|
2008-04-14 05:30:24 +08:00
|
|
|
|
RecordDecl *Decl = (RecordDecl*)classdecl;
|
|
|
|
|
QualType BaseType = Context.getTypeDeclType((TypeDecl*)basetype);
|
|
|
|
|
|
|
|
|
|
// Base specifiers must be record types.
|
2008-11-19 13:08:23 +08:00
|
|
|
|
if (!BaseType->isRecordType())
|
|
|
|
|
return Diag(BaseLoc, diag::err_base_must_be_class) << SpecifierRange;
|
2008-04-14 05:30:24 +08:00
|
|
|
|
|
|
|
|
|
// C++ [class.union]p1:
|
|
|
|
|
// A union shall not be used as a base class.
|
2008-11-19 13:08:23 +08:00
|
|
|
|
if (BaseType->isUnionType())
|
|
|
|
|
return Diag(BaseLoc, diag::err_union_as_base_class) << SpecifierRange;
|
2008-04-14 05:30:24 +08:00
|
|
|
|
|
|
|
|
|
// C++ [class.union]p1:
|
|
|
|
|
// A union shall not have base classes.
|
2008-11-19 13:08:23 +08:00
|
|
|
|
if (Decl->isUnion())
|
|
|
|
|
return Diag(Decl->getLocation(), diag::err_base_clause_on_union)
|
|
|
|
|
<< SpecifierRange;
|
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.
|
2008-11-19 13:08:23 +08:00
|
|
|
|
if (BaseType->isIncompleteType())
|
|
|
|
|
return Diag(BaseLoc, diag::err_incomplete_base_class) << SpecifierRange;
|
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())
|
2008-11-06 23:59:35 +08:00
|
|
|
|
cast<CXXRecordDecl>(Decl)->setPolymorphic(true);
|
|
|
|
|
|
2008-10-23 01:49:05 +08:00
|
|
|
|
// Create the base specifier.
|
2008-10-24 02:13:27 +08:00
|
|
|
|
return new CXXBaseSpecifier(SpecifierRange, Virtual,
|
|
|
|
|
BaseType->isClassType(), Access, BaseType);
|
2008-10-23 01:49:05 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/// ActOnBaseSpecifiers - Attach the given base specifiers to the
|
|
|
|
|
/// class, after checking whether there are any duplicate base
|
|
|
|
|
/// classes.
|
|
|
|
|
void Sema::ActOnBaseSpecifiers(DeclTy *ClassDecl, BaseTy **Bases,
|
|
|
|
|
unsigned NumBases) {
|
|
|
|
|
if (NumBases == 0)
|
|
|
|
|
return;
|
|
|
|
|
|
|
|
|
|
// 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
|
|
|
|
CXXBaseSpecifier **BaseSpecs = (CXXBaseSpecifier **)Bases;
|
|
|
|
|
unsigned NumGoodBases = 0;
|
|
|
|
|
for (unsigned idx = 0; idx < NumBases; ++idx) {
|
2008-10-23 01:49:05 +08:00
|
|
|
|
QualType NewBaseType
|
2008-10-24 02:13:27 +08:00
|
|
|
|
= Context.getCanonicalType(BaseSpecs[idx]->getType());
|
|
|
|
|
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.
|
2008-10-24 02:13:27 +08:00
|
|
|
|
Diag(BaseSpecs[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()
|
2008-11-19 13:08:23 +08:00
|
|
|
|
<< BaseSpecs[idx]->getSourceRange();
|
2008-10-24 02:13:27 +08:00
|
|
|
|
|
|
|
|
|
// Delete the duplicate base class specifier; we're going to
|
|
|
|
|
// overwrite its pointer later.
|
|
|
|
|
delete BaseSpecs[idx];
|
2008-10-23 01:49:05 +08:00
|
|
|
|
} else {
|
|
|
|
|
// Okay, add this new base class.
|
2008-10-24 02:13:27 +08:00
|
|
|
|
KnownBaseTypes[NewBaseType] = BaseSpecs[idx];
|
|
|
|
|
BaseSpecs[NumGoodBases++] = BaseSpecs[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.
|
|
|
|
|
CXXRecordDecl *Decl = (CXXRecordDecl*)ClassDecl;
|
2008-10-24 02:13:27 +08:00
|
|
|
|
Decl->setBases(BaseSpecs, NumGoodBases);
|
|
|
|
|
|
|
|
|
|
// Delete the remaining (good) base class specifiers, since their
|
|
|
|
|
// data has been copied into the CXXRecordDecl.
|
|
|
|
|
for (unsigned idx = 0; idx < NumGoodBases; ++idx)
|
|
|
|
|
delete BaseSpecs[idx];
|
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
|
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
|
|
|
|
|
|
/// ActOnStartCXXClassDef - This is called at the start of a class/struct/union
|
|
|
|
|
/// definition, when on C++.
|
|
|
|
|
void Sema::ActOnStartCXXClassDef(Scope *S, DeclTy *D, SourceLocation LBrace) {
|
2008-10-31 17:07:45 +08:00
|
|
|
|
CXXRecordDecl *Dcl = cast<CXXRecordDecl>(static_cast<Decl *>(D));
|
2008-12-12 00:49:14 +08:00
|
|
|
|
PushDeclContext(S, Dcl);
|
2008-07-01 18:37:29 +08:00
|
|
|
|
FieldCollector->StartClass();
|
2008-10-31 17:07:45 +08:00
|
|
|
|
|
|
|
|
|
if (Dcl->getIdentifier()) {
|
|
|
|
|
// C++ [class]p2:
|
|
|
|
|
// [...] The class-name is also inserted into the scope of the
|
|
|
|
|
// class itself; this is known as the injected-class-name. For
|
|
|
|
|
// purposes of access checking, the injected-class-name is treated
|
|
|
|
|
// as if it were a public member name.
|
2008-11-10 22:41:22 +08:00
|
|
|
|
PushOnScopeChains(Dcl, S);
|
2008-10-31 17:07:45 +08:00
|
|
|
|
}
|
2008-07-01 18:37:29 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/// 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
|
|
|
|
|
/// any. 'LastInGroup' is non-null for cases where one declspec has multiple
|
|
|
|
|
/// declarators on it.
|
|
|
|
|
///
|
2008-12-17 05:30:33 +08:00
|
|
|
|
/// FIXME: The note below is out-of-date.
|
2008-07-01 18:37:29 +08:00
|
|
|
|
/// NOTE: Because of CXXFieldDecl's inability to be chained like ScopedDecls, if
|
|
|
|
|
/// an instance field is declared, a new CXXFieldDecl is created but the method
|
|
|
|
|
/// does *not* return it; it returns LastInGroup instead. The other C++ members
|
|
|
|
|
/// (which are all ScopedDecls) are returned after appending them to
|
|
|
|
|
/// LastInGroup.
|
|
|
|
|
Sema::DeclTy *
|
|
|
|
|
Sema::ActOnCXXMemberDeclarator(Scope *S, AccessSpecifier AS, Declarator &D,
|
|
|
|
|
ExprTy *BW, ExprTy *InitExpr,
|
|
|
|
|
DeclTy *LastInGroup) {
|
|
|
|
|
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 &&
|
|
|
|
|
D.getDeclSpec().getTypeSpecType() == DeclSpec::TST_typedef &&
|
|
|
|
|
D.getNumTypeObjects() == 0) {
|
2008-10-09 06:20:31 +08:00
|
|
|
|
// Check also for this case:
|
|
|
|
|
//
|
|
|
|
|
// typedef int f();
|
|
|
|
|
// f a;
|
|
|
|
|
//
|
|
|
|
|
Decl *TD = static_cast<Decl *>(DS.getTypeRep());
|
|
|
|
|
isFunc = Context.getTypeDeclType(cast<TypeDecl>(TD))->isFunctionType();
|
|
|
|
|
}
|
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;
|
|
|
|
|
bool InvalidDecl = false;
|
|
|
|
|
|
|
|
|
|
if (isInstField)
|
2008-12-12 00:49:14 +08:00
|
|
|
|
Member = static_cast<Decl*>(ActOnField(S, cast<CXXRecordDecl>(CurContext),
|
|
|
|
|
Loc, D, BitWidth));
|
2008-07-01 18:37:29 +08:00
|
|
|
|
else
|
2008-08-06 00:28:08 +08:00
|
|
|
|
Member = static_cast<Decl*>(ActOnDeclarator(S, D, LastInGroup));
|
2008-07-01 18:37:29 +08:00
|
|
|
|
|
|
|
|
|
if (!Member) return LastInGroup;
|
|
|
|
|
|
2008-11-18 06:58:34 +08:00
|
|
|
|
assert((Name || isInstField) && "No identifier for non-field ?");
|
2008-07-01 18:37:29 +08:00
|
|
|
|
|
|
|
|
|
// set/getAccess is not part of Decl's interface to avoid bloating it with C++
|
|
|
|
|
// specific methods. Use a wrapper class that can be used with all C++ class
|
|
|
|
|
// member decls.
|
|
|
|
|
CXXClassMemberWrapper(Member).setAccess(AS);
|
|
|
|
|
|
2008-11-06 00:20:31 +08:00
|
|
|
|
// C++ [dcl.init.aggr]p1:
|
|
|
|
|
// An aggregate is an array or a class (clause 9) with [...] no
|
|
|
|
|
// private or protected non-static data members (clause 11).
|
|
|
|
|
if (isInstField && (AS == AS_private || AS == AS_protected))
|
|
|
|
|
cast<CXXRecordDecl>(CurContext)->setAggregate(false);
|
|
|
|
|
|
2008-11-06 23:59:35 +08:00
|
|
|
|
if (DS.isVirtualSpecified()) {
|
|
|
|
|
if (!isFunc || DS.getStorageClassSpec() == DeclSpec::SCS_static) {
|
|
|
|
|
Diag(DS.getVirtualSpecLoc(), diag::err_virtual_non_function);
|
|
|
|
|
InvalidDecl = true;
|
|
|
|
|
} else {
|
|
|
|
|
CXXRecordDecl *CurClass = cast<CXXRecordDecl>(CurContext);
|
|
|
|
|
CurClass->setAggregate(false);
|
|
|
|
|
CurClass->setPolymorphic(true);
|
|
|
|
|
}
|
|
|
|
|
}
|
2008-11-06 00:20:31 +08:00
|
|
|
|
|
2008-07-01 18:37:29 +08:00
|
|
|
|
if (BitWidth) {
|
|
|
|
|
// C++ 9.6p2: Only when declaring an unnamed bit-field may the
|
|
|
|
|
// constant-expression be a value equal to zero.
|
|
|
|
|
// FIXME: Check this.
|
|
|
|
|
|
|
|
|
|
if (D.isFunctionDeclarator()) {
|
|
|
|
|
// FIXME: Emit diagnostic about only constructors taking base initializers
|
|
|
|
|
// or something similar, when constructor support is in place.
|
2008-11-19 13:08:23 +08:00
|
|
|
|
Diag(Loc, diag::err_not_bitfield_type)
|
2008-12-07 04:05:35 +08:00
|
|
|
|
<< Name << BitWidth->getSourceRange();
|
2008-07-01 18:37:29 +08:00
|
|
|
|
InvalidDecl = true;
|
|
|
|
|
|
2008-10-09 06:20:31 +08:00
|
|
|
|
} else if (isInstField) {
|
2008-07-01 18:37:29 +08:00
|
|
|
|
// C++ 9.6p3: A bit-field shall have integral or enumeration type.
|
2008-10-09 06:20:31 +08:00
|
|
|
|
if (!cast<FieldDecl>(Member)->getType()->isIntegralType()) {
|
2008-11-19 13:08:23 +08:00
|
|
|
|
Diag(Loc, diag::err_not_integral_type_bitfield)
|
2008-12-07 04:05:35 +08:00
|
|
|
|
<< Name << BitWidth->getSourceRange();
|
2008-07-01 18:37:29 +08:00
|
|
|
|
InvalidDecl = true;
|
|
|
|
|
}
|
|
|
|
|
|
2008-10-09 06:20:31 +08:00
|
|
|
|
} else if (isa<FunctionDecl>(Member)) {
|
|
|
|
|
// A function typedef ("typedef int f(); f a;").
|
|
|
|
|
// C++ 9.6p3: A bit-field shall have integral or enumeration type.
|
2008-11-19 13:08:23 +08:00
|
|
|
|
Diag(Loc, diag::err_not_integral_type_bitfield)
|
2008-12-07 04:05:35 +08:00
|
|
|
|
<< Name << BitWidth->getSourceRange();
|
2008-10-09 06:20:31 +08:00
|
|
|
|
InvalidDecl = true;
|
|
|
|
|
|
2008-07-01 18:37:29 +08:00
|
|
|
|
} else if (isa<TypedefDecl>(Member)) {
|
|
|
|
|
// "cannot declare 'A' to be a bit-field type"
|
2008-11-19 13:08:23 +08:00
|
|
|
|
Diag(Loc, diag::err_not_bitfield_type)
|
2008-12-07 04:05:35 +08:00
|
|
|
|
<< Name << BitWidth->getSourceRange();
|
2008-07-01 18:37:29 +08:00
|
|
|
|
InvalidDecl = true;
|
|
|
|
|
|
|
|
|
|
} else {
|
|
|
|
|
assert(isa<CXXClassVarDecl>(Member) &&
|
|
|
|
|
"Didn't we cover all member kinds?");
|
|
|
|
|
// C++ 9.6p3: A bit-field shall not be a static member.
|
|
|
|
|
// "static member 'A' cannot be a bit-field"
|
2008-11-19 13:08:23 +08:00
|
|
|
|
Diag(Loc, diag::err_static_not_bitfield)
|
2008-12-07 04:05:35 +08:00
|
|
|
|
<< Name << BitWidth->getSourceRange();
|
2008-07-01 18:37:29 +08:00
|
|
|
|
InvalidDecl = true;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (Init) {
|
|
|
|
|
// C++ 9.2p4: A member-declarator can contain a constant-initializer only
|
|
|
|
|
// if it declares a static member of const integral or const enumeration
|
|
|
|
|
// type.
|
2008-07-27 06:17:49 +08:00
|
|
|
|
if (CXXClassVarDecl *CVD = dyn_cast<CXXClassVarDecl>(Member)) {
|
|
|
|
|
// ...static member of...
|
2008-07-01 18:37:29 +08:00
|
|
|
|
CVD->setInit(Init);
|
|
|
|
|
// ...const integral or const enumeration type.
|
2008-07-27 06:17:49 +08:00
|
|
|
|
if (Context.getCanonicalType(CVD->getType()).isConstQualified() &&
|
|
|
|
|
CVD->getType()->isIntegralType()) {
|
|
|
|
|
// constant-initializer
|
|
|
|
|
if (CheckForConstantInitializer(Init, CVD->getType()))
|
2008-07-01 18:37:29 +08:00
|
|
|
|
InvalidDecl = true;
|
|
|
|
|
|
|
|
|
|
} else {
|
|
|
|
|
// not const integral.
|
2008-11-20 14:06:08 +08:00
|
|
|
|
Diag(Loc, diag::err_member_initialization)
|
2008-12-07 04:05:35 +08:00
|
|
|
|
<< Name << Init->getSourceRange();
|
2008-07-01 18:37:29 +08:00
|
|
|
|
InvalidDecl = true;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
} else {
|
|
|
|
|
// not static member.
|
2008-11-20 14:06:08 +08:00
|
|
|
|
Diag(Loc, diag::err_member_initialization)
|
2008-12-07 04:05:35 +08:00
|
|
|
|
<< Name << Init->getSourceRange();
|
2008-07-01 18:37:29 +08:00
|
|
|
|
InvalidDecl = true;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (InvalidDecl)
|
|
|
|
|
Member->setInvalidDecl();
|
|
|
|
|
|
|
|
|
|
if (isInstField) {
|
2008-12-12 00:49:14 +08:00
|
|
|
|
FieldCollector->Add(cast<FieldDecl>(Member));
|
2008-07-01 18:37:29 +08:00
|
|
|
|
return LastInGroup;
|
|
|
|
|
}
|
|
|
|
|
return Member;
|
|
|
|
|
}
|
|
|
|
|
|
2008-11-05 12:29:56 +08:00
|
|
|
|
/// ActOnMemInitializer - Handle a C++ member initializer.
|
|
|
|
|
Sema::MemInitResult
|
|
|
|
|
Sema::ActOnMemInitializer(DeclTy *ConstructorD,
|
|
|
|
|
Scope *S,
|
|
|
|
|
IdentifierInfo *MemberOrBase,
|
|
|
|
|
SourceLocation IdLoc,
|
|
|
|
|
SourceLocation LParenLoc,
|
|
|
|
|
ExprTy **Args, unsigned NumArgs,
|
|
|
|
|
SourceLocation *CommaLocs,
|
|
|
|
|
SourceLocation RParenLoc) {
|
|
|
|
|
CXXConstructorDecl *Constructor
|
|
|
|
|
= dyn_cast<CXXConstructorDecl>((Decl*)ConstructorD);
|
|
|
|
|
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;
|
|
|
|
|
DeclContext::lookup_result Result = ClassDecl->lookup(Context, MemberOrBase);
|
|
|
|
|
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.
|
2008-11-09 00:45:02 +08:00
|
|
|
|
TypeTy *BaseTy = isTypeName(*MemberOrBase, 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
|
|
|
|
|
|
|
|
|
QualType BaseType = Context.getTypeDeclType((TypeDecl *)BaseTy);
|
|
|
|
|
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.
|
|
|
|
|
// FIXME: We might be able to short-circuit this if we know in
|
|
|
|
|
// advance that there are no virtual bases.
|
|
|
|
|
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);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2008-07-01 18:37:29 +08:00
|
|
|
|
void Sema::ActOnFinishCXXMemberSpecification(Scope* S, SourceLocation RLoc,
|
|
|
|
|
DeclTy *TagDecl,
|
|
|
|
|
SourceLocation LBrac,
|
|
|
|
|
SourceLocation RBrac) {
|
|
|
|
|
ActOnFields(S, RLoc, TagDecl,
|
|
|
|
|
(DeclTy**)FieldCollector->getCurFields(),
|
2008-10-03 10:03:53 +08:00
|
|
|
|
FieldCollector->getCurNumFields(), LBrac, RBrac, 0);
|
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);
|
|
|
|
|
|
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);
|
2008-12-16 05:24:18 +08:00
|
|
|
|
ClassDecl->addDecl(Context, DefaultCon);
|
|
|
|
|
|
|
|
|
|
// 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&.
|
|
|
|
|
for (CXXRecordDecl::field_iterator Field = ClassDecl->field_begin();
|
|
|
|
|
HasConstCopyConstructor && Field != ClassDecl->field_end(); ++Field) {
|
|
|
|
|
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);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Otherwise, the implicitly declared copy constructor will have
|
|
|
|
|
// the form
|
|
|
|
|
//
|
|
|
|
|
// X::X(X&)
|
|
|
|
|
QualType ArgType = Context.getTypeDeclType(ClassDecl);
|
|
|
|
|
if (HasConstCopyConstructor)
|
|
|
|
|
ArgType = ArgType.withConst();
|
|
|
|
|
ArgType = Context.getReferenceType(ArgType);
|
|
|
|
|
|
|
|
|
|
// 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);
|
|
|
|
|
|
|
|
|
|
// Add the parameter to the constructor.
|
|
|
|
|
ParmVarDecl *FromParam = ParmVarDecl::Create(Context, CopyConstructor,
|
|
|
|
|
ClassDecl->getLocation(),
|
|
|
|
|
/*IdentifierInfo=*/0,
|
|
|
|
|
ArgType, VarDecl::None, 0, 0);
|
|
|
|
|
CopyConstructor->setParams(&FromParam, 1);
|
|
|
|
|
|
2008-12-16 05:24:18 +08:00
|
|
|
|
ClassDecl->addedConstructor(Context, CopyConstructor);
|
|
|
|
|
DeclContext::lookup_result Lookup = ClassDecl->lookup(Context, Name);
|
|
|
|
|
if (Lookup.first == Lookup.second
|
|
|
|
|
|| (!isa<CXXConstructorDecl>(*Lookup.first) &&
|
|
|
|
|
!isa<OverloadedFunctionDecl>(*Lookup.first)))
|
|
|
|
|
ClassDecl->addDecl(Context, CopyConstructor);
|
|
|
|
|
else {
|
|
|
|
|
OverloadedFunctionDecl *Ovl
|
|
|
|
|
= dyn_cast<OverloadedFunctionDecl>(*Lookup.first);
|
|
|
|
|
if (!Ovl) {
|
|
|
|
|
Ovl = OverloadedFunctionDecl::Create(Context, ClassDecl, Name);
|
|
|
|
|
Ovl->addOverload(cast<CXXConstructorDecl>(*Lookup.first));
|
|
|
|
|
ClassDecl->insert(Context, Ovl);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
Ovl->addOverload(CopyConstructor);
|
|
|
|
|
ClassDecl->addDecl(Context, CopyConstructor, false);
|
|
|
|
|
}
|
2008-11-04 01:51:48 +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);
|
2008-12-16 05:24:18 +08:00
|
|
|
|
ClassDecl->addDecl(Context, Destructor);
|
2008-11-06 04:51:48 +08:00
|
|
|
|
}
|
|
|
|
|
|
2008-11-04 01:51:48 +08:00
|
|
|
|
// FIXME: Implicit copy assignment operator
|
|
|
|
|
}
|
|
|
|
|
|
2008-08-09 08:39:29 +08:00
|
|
|
|
void Sema::ActOnFinishCXXClassDef(DeclTy *D) {
|
2008-08-09 08:58:37 +08:00
|
|
|
|
CXXRecordDecl *Rec = cast<CXXRecordDecl>(static_cast<Decl *>(D));
|
2008-07-01 18:37:29 +08:00
|
|
|
|
FieldCollector->FinishClass();
|
2008-11-04 01:51:48 +08:00
|
|
|
|
AddImplicitlyDeclaredMembersToClass(Rec);
|
2008-07-01 18:37:29 +08:00
|
|
|
|
PopDeclContext();
|
2008-08-09 08:58:37 +08:00
|
|
|
|
|
|
|
|
|
// Everything, including inline method definitions, have been parsed.
|
|
|
|
|
// Let the consumer know of the new TagDecl definition.
|
|
|
|
|
Consumer.HandleTagDeclDefinition(Rec);
|
2008-07-01 18:37:29 +08:00
|
|
|
|
}
|
2008-04-27 21:50:30 +08:00
|
|
|
|
|
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.
|
|
|
|
|
void Sema::ActOnStartDelayedCXXMethodDeclaration(Scope *S, DeclTy *Method) {
|
|
|
|
|
CXXScopeSpec SS;
|
|
|
|
|
SS.setScopeRep(((FunctionDecl*)Method)->getDeclContext());
|
|
|
|
|
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.
|
|
|
|
|
void Sema::ActOnDelayedCXXMethodParameter(Scope *S, DeclTy *ParamD) {
|
|
|
|
|
ParmVarDecl *Param = (ParmVarDecl*)ParamD;
|
|
|
|
|
S->AddDecl(Param);
|
|
|
|
|
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.
|
|
|
|
|
void Sema::ActOnFinishDelayedCXXMethodDeclaration(Scope *S, DeclTy *MethodD) {
|
|
|
|
|
FunctionDecl *Method = (FunctionDecl*)MethodD;
|
|
|
|
|
CXXScopeSpec SS;
|
|
|
|
|
SS.setScopeRep(Method->getDeclContext());
|
|
|
|
|
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.
|
|
|
|
|
if (CXXConstructorDecl *Constructor = dyn_cast<CXXConstructorDecl>(Method)) {
|
|
|
|
|
if (CheckConstructor(Constructor))
|
|
|
|
|
Constructor->setInvalidDecl();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// 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
|
|
|
|
|
/// 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 constructor.
|
|
|
|
|
bool Sema::CheckConstructorDeclarator(Declarator &D, QualType &R,
|
|
|
|
|
FunctionDecl::StorageClass& SC) {
|
|
|
|
|
bool isVirtual = D.getDeclSpec().isVirtualSpecified();
|
|
|
|
|
bool isInvalid = false;
|
|
|
|
|
|
|
|
|
|
// 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) {
|
2008-11-19 13:08:23 +08:00
|
|
|
|
Diag(D.getIdentifierLoc(), diag::err_constructor_cannot_be)
|
|
|
|
|
<< "virtual" << SourceRange(D.getDeclSpec().getVirtualSpecLoc())
|
|
|
|
|
<< SourceRange(D.getIdentifierLoc());
|
2008-11-06 04:51:48 +08:00
|
|
|
|
isInvalid = true;
|
|
|
|
|
}
|
|
|
|
|
if (SC == FunctionDecl::Static) {
|
2008-11-19 13:08:23 +08:00
|
|
|
|
Diag(D.getIdentifierLoc(), diag::err_constructor_cannot_be)
|
|
|
|
|
<< "static" << SourceRange(D.getDeclSpec().getStorageClassSpecLoc())
|
|
|
|
|
<< SourceRange(D.getIdentifierLoc());
|
2008-11-06 04:51:48 +08:00
|
|
|
|
isInvalid = true;
|
|
|
|
|
SC = FunctionDecl::None;
|
|
|
|
|
}
|
|
|
|
|
if (D.getDeclSpec().hasTypeSpecifier()) {
|
|
|
|
|
// Constructors don't have return types, but the parser will
|
|
|
|
|
// happily parse something like:
|
|
|
|
|
//
|
|
|
|
|
// class X {
|
|
|
|
|
// float X(float);
|
|
|
|
|
// };
|
|
|
|
|
//
|
|
|
|
|
// The return type will be eliminated later.
|
2008-11-19 13:08:23 +08:00
|
|
|
|
Diag(D.getIdentifierLoc(), diag::err_constructor_return_type)
|
|
|
|
|
<< SourceRange(D.getDeclSpec().getTypeSpecTypeLoc())
|
|
|
|
|
<< SourceRange(D.getIdentifierLoc());
|
2008-11-06 04:51:48 +08:00
|
|
|
|
}
|
|
|
|
|
if (R->getAsFunctionTypeProto()->getTypeQuals() != 0) {
|
|
|
|
|
DeclaratorChunk::FunctionTypeInfo &FTI = D.getTypeObject(0).Fun;
|
|
|
|
|
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.
|
|
|
|
|
const FunctionTypeProto *Proto = R->getAsFunctionTypeProto();
|
|
|
|
|
R = Context.getFunctionType(Context.VoidTy, Proto->arg_type_begin(),
|
|
|
|
|
Proto->getNumArgs(),
|
|
|
|
|
Proto->isVariadic(),
|
|
|
|
|
0);
|
|
|
|
|
|
|
|
|
|
return isInvalid;
|
|
|
|
|
}
|
|
|
|
|
|
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.
|
|
|
|
|
bool Sema::CheckConstructor(CXXConstructorDecl *Constructor) {
|
|
|
|
|
if (Constructor->isInvalidDecl())
|
|
|
|
|
return true;
|
|
|
|
|
|
|
|
|
|
CXXRecordDecl *ClassDecl = cast<CXXRecordDecl>(Constructor->getDeclContext());
|
|
|
|
|
bool Invalid = false;
|
|
|
|
|
|
|
|
|
|
// 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.
|
|
|
|
|
if ((Constructor->getNumParams() == 1) ||
|
|
|
|
|
(Constructor->getNumParams() > 1 &&
|
|
|
|
|
Constructor->getParamDecl(1)->getDefaultArg() != 0)) {
|
|
|
|
|
QualType ParamType = Constructor->getParamDecl(0)->getType();
|
|
|
|
|
QualType ClassTy = Context.getTagDeclType(ClassDecl);
|
|
|
|
|
if (Context.getCanonicalType(ParamType).getUnqualifiedType() == ClassTy) {
|
|
|
|
|
Diag(Constructor->getLocation(), diag::err_constructor_byvalue_arg)
|
|
|
|
|
<< SourceRange(Constructor->getParamDecl(0)->getLocation());
|
|
|
|
|
Invalid = true;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Notify the class that we've added a constructor.
|
|
|
|
|
ClassDecl->addedConstructor(Context, Constructor);
|
|
|
|
|
|
|
|
|
|
return Invalid;
|
|
|
|
|
}
|
|
|
|
|
|
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
|
|
|
|
|
/// 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 destructor.
|
|
|
|
|
bool Sema::CheckDestructorDeclarator(Declarator &D, QualType &R,
|
|
|
|
|
FunctionDecl::StorageClass& SC) {
|
|
|
|
|
bool isInvalid = false;
|
|
|
|
|
|
|
|
|
|
// 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.
|
|
|
|
|
TypeDecl *DeclaratorTypeD = (TypeDecl *)D.getDeclaratorIdType();
|
|
|
|
|
if (const TypedefDecl *TypedefD = dyn_cast<TypedefDecl>(DeclaratorTypeD)) {
|
2008-11-19 13:08:23 +08:00
|
|
|
|
Diag(D.getIdentifierLoc(), diag::err_destructor_typedef_name)
|
2008-11-24 13:29:24 +08:00
|
|
|
|
<< TypedefD->getDeclName();
|
2008-11-10 22:41:22 +08:00
|
|
|
|
isInvalid = true;
|
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) {
|
2008-11-19 13:08:23 +08:00
|
|
|
|
Diag(D.getIdentifierLoc(), diag::err_destructor_cannot_be)
|
|
|
|
|
<< "static" << SourceRange(D.getDeclSpec().getStorageClassSpecLoc())
|
|
|
|
|
<< SourceRange(D.getIdentifierLoc());
|
2008-11-06 04:51:48 +08:00
|
|
|
|
isInvalid = true;
|
|
|
|
|
SC = FunctionDecl::None;
|
|
|
|
|
}
|
|
|
|
|
if (D.getDeclSpec().hasTypeSpecifier()) {
|
|
|
|
|
// 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
|
|
|
|
}
|
|
|
|
|
if (R->getAsFunctionTypeProto()->getTypeQuals() != 0) {
|
|
|
|
|
DeclaratorChunk::FunctionTypeInfo &FTI = D.getTypeObject(0).Fun;
|
|
|
|
|
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());
|
2008-11-06 04:51:48 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Make sure we don't have any parameters.
|
|
|
|
|
if (R->getAsFunctionTypeProto()->getNumArgs() > 0) {
|
|
|
|
|
Diag(D.getIdentifierLoc(), diag::err_destructor_with_params);
|
|
|
|
|
|
|
|
|
|
// Delete the parameters.
|
|
|
|
|
DeclaratorChunk::FunctionTypeInfo &FTI = D.getTypeObject(0).Fun;
|
|
|
|
|
if (FTI.NumArgs) {
|
|
|
|
|
delete [] FTI.ArgInfo;
|
|
|
|
|
FTI.NumArgs = 0;
|
|
|
|
|
FTI.ArgInfo = 0;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Make sure the destructor isn't variadic.
|
|
|
|
|
if (R->getAsFunctionTypeProto()->isVariadic())
|
|
|
|
|
Diag(D.getIdentifierLoc(), diag::err_destructor_variadic);
|
|
|
|
|
|
|
|
|
|
// 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.
|
|
|
|
|
R = Context.getFunctionType(Context.VoidTy, 0, 0, false, 0);
|
|
|
|
|
|
|
|
|
|
return isInvalid;
|
|
|
|
|
}
|
|
|
|
|
|
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.
|
|
|
|
|
bool Sema::CheckConversionDeclarator(Declarator &D, QualType &R,
|
|
|
|
|
FunctionDecl::StorageClass& SC) {
|
|
|
|
|
bool isInvalid = false;
|
|
|
|
|
|
|
|
|
|
// 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) {
|
2008-11-19 13:08:23 +08:00
|
|
|
|
Diag(D.getIdentifierLoc(), diag::err_conv_function_not_member)
|
|
|
|
|
<< "static" << SourceRange(D.getDeclSpec().getStorageClassSpecLoc())
|
|
|
|
|
<< SourceRange(D.getIdentifierLoc());
|
2008-11-08 04:08:42 +08:00
|
|
|
|
isInvalid = true;
|
|
|
|
|
SC = FunctionDecl::None;
|
|
|
|
|
}
|
|
|
|
|
if (D.getDeclSpec().hasTypeSpecifier()) {
|
|
|
|
|
// 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.
|
|
|
|
|
if (R->getAsFunctionTypeProto()->getNumArgs() > 0) {
|
|
|
|
|
Diag(D.getIdentifierLoc(), diag::err_conv_function_with_params);
|
|
|
|
|
|
|
|
|
|
// Delete the parameters.
|
|
|
|
|
DeclaratorChunk::FunctionTypeInfo &FTI = D.getTypeObject(0).Fun;
|
|
|
|
|
if (FTI.NumArgs) {
|
|
|
|
|
delete [] FTI.ArgInfo;
|
|
|
|
|
FTI.NumArgs = 0;
|
|
|
|
|
FTI.ArgInfo = 0;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Make sure the conversion function isn't variadic.
|
|
|
|
|
if (R->getAsFunctionTypeProto()->isVariadic())
|
|
|
|
|
Diag(D.getIdentifierLoc(), diag::err_conv_function_variadic);
|
|
|
|
|
|
|
|
|
|
// 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);
|
|
|
|
|
} else if (ConvType->isFunctionType()) {
|
|
|
|
|
Diag(D.getIdentifierLoc(), diag::err_conv_function_to_function);
|
|
|
|
|
ConvType = Context.getPointerType(ConvType);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// 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,
|
|
|
|
|
R->getAsFunctionTypeProto()->getTypeQuals());
|
|
|
|
|
|
|
|
|
|
return isInvalid;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/// 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.
|
|
|
|
|
Sema::DeclTy *Sema::ActOnConversionDeclarator(CXXConversionDecl *Conversion) {
|
|
|
|
|
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());
|
|
|
|
|
OverloadedFunctionDecl *Conversions = ClassDecl->getConversionFunctions();
|
|
|
|
|
for (OverloadedFunctionDecl::function_iterator Func
|
|
|
|
|
= Conversions->function_begin();
|
|
|
|
|
Func != Conversions->function_end(); ++Func) {
|
|
|
|
|
CXXConversionDecl *OtherConv = cast<CXXConversionDecl>(*Func);
|
|
|
|
|
if (ConvType == Context.getCanonicalType(OtherConv->getConversionType())) {
|
|
|
|
|
Diag(Conversion->getLocation(), diag::err_conv_function_redeclared);
|
|
|
|
|
Diag(OtherConv->getLocation(),
|
|
|
|
|
OtherConv->isThisDeclarationADefinition()?
|
2008-11-24 07:12:31 +08:00
|
|
|
|
diag::note_previous_definition
|
|
|
|
|
: diag::note_previous_declaration);
|
2008-11-08 04:08:42 +08:00
|
|
|
|
Conversion->setInvalidDecl();
|
|
|
|
|
return (DeclTy *)Conversion;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// 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.
|
|
|
|
|
// FIXME: Suppress this warning if the conversion function ends up
|
|
|
|
|
// being a virtual function that overrides a virtual function in a
|
|
|
|
|
// base class.
|
|
|
|
|
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
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
ClassDecl->addConversionFunction(Context, Conversion);
|
|
|
|
|
|
|
|
|
|
return (DeclTy *)Conversion;
|
|
|
|
|
}
|
|
|
|
|
|
2008-04-27 21:50:30 +08:00
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
|
// Namespace Handling
|
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
|
|
|
|
|
|
/// ActOnStartNamespaceDef - This is called at the start of a namespace
|
|
|
|
|
/// definition.
|
|
|
|
|
Sema::DeclTy *Sema::ActOnStartNamespaceDef(Scope *NamespcScope,
|
|
|
|
|
SourceLocation IdentLoc,
|
|
|
|
|
IdentifierInfo *II,
|
|
|
|
|
SourceLocation LBrace) {
|
|
|
|
|
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.
|
|
|
|
|
|
|
|
|
|
Decl *PrevDecl =
|
2008-12-12 00:49:14 +08:00
|
|
|
|
LookupDecl(II, Decl::IDNS_Tag | Decl::IDNS_Ordinary, DeclRegionScope, 0,
|
|
|
|
|
/*enableLazyBuiltinCreation=*/false,
|
|
|
|
|
/*LookupInParent=*/false);
|
|
|
|
|
|
|
|
|
|
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.
|
|
|
|
|
if (DeclRegionScope->isDeclScope(OrigNS)) {
|
2008-12-12 04:41:00 +08:00
|
|
|
|
IdResolver.RemoveDecl(OrigNS);
|
|
|
|
|
DeclRegionScope->RemoveDecl(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.
|
2008-12-12 00:49:14 +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.
|
|
|
|
|
PushDeclContext(NamespcScope, Namespc);
|
2008-04-27 21:50:30 +08:00
|
|
|
|
return Namespc;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/// ActOnFinishNamespaceDef - This callback is called after a namespace is
|
|
|
|
|
/// exited. Decl is the DeclTy returned by ActOnStartNamespaceDef.
|
|
|
|
|
void Sema::ActOnFinishNamespaceDef(DeclTy *D, SourceLocation RBrace) {
|
|
|
|
|
Decl *Dcl = static_cast<Decl *>(D);
|
|
|
|
|
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
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/// AddCXXDirectInitializerToDecl - This action is called immediately after
|
|
|
|
|
/// ActOnDeclarator, when a C++ direct initializer is present.
|
|
|
|
|
/// e.g: "int x(1);"
|
|
|
|
|
void Sema::AddCXXDirectInitializerToDecl(DeclTy *Dcl, SourceLocation LParenLoc,
|
|
|
|
|
ExprTy **ExprTys, unsigned NumExprs,
|
|
|
|
|
SourceLocation *CommaLocs,
|
|
|
|
|
SourceLocation RParenLoc) {
|
|
|
|
|
assert(NumExprs != 0 && ExprTys && "missing expressions");
|
2008-10-07 07:08:37 +08:00
|
|
|
|
Decl *RealDecl = static_cast<Decl *>(Dcl);
|
2008-10-07 01:10:33 +08:00
|
|
|
|
|
|
|
|
|
// If there is no declaration, there was an error parsing it. Just ignore
|
|
|
|
|
// the initializer.
|
|
|
|
|
if (RealDecl == 0) {
|
2008-10-07 04:35:04 +08:00
|
|
|
|
for (unsigned i = 0; i != NumExprs; ++i)
|
2008-10-07 01:10:33 +08:00
|
|
|
|
delete static_cast<Expr *>(ExprTys[i]);
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
VarDecl *VDecl = dyn_cast<VarDecl>(RealDecl);
|
|
|
|
|
if (!VDecl) {
|
|
|
|
|
Diag(RealDecl->getLocation(), diag::err_illegal_initializer);
|
|
|
|
|
RealDecl->setInvalidDecl();
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
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
|
|
|
|
|
|
|
|
|
if (VDecl->getType()->isRecordType()) {
|
2008-11-04 04:45:27 +08:00
|
|
|
|
CXXConstructorDecl *Constructor
|
2008-11-05 23:29:30 +08:00
|
|
|
|
= PerformInitializationByConstructor(DeclInitType,
|
|
|
|
|
(Expr **)ExprTys, NumExprs,
|
|
|
|
|
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);
|
2008-11-04 04:45:27 +08:00
|
|
|
|
if (!Constructor) {
|
|
|
|
|
RealDecl->setInvalidDecl();
|
|
|
|
|
}
|
2008-11-05 23:29:30 +08:00
|
|
|
|
|
|
|
|
|
// Let clients know that initialization was done with a direct
|
|
|
|
|
// initializer.
|
|
|
|
|
VDecl->setCXXDirectInitializer(true);
|
|
|
|
|
|
|
|
|
|
// FIXME: Add ExprTys and Constructor to the RealDecl as part of
|
|
|
|
|
// the initializer.
|
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.
|
2008-12-14 00:23:55 +08:00
|
|
|
|
AddInitializerToDecl(Dcl, ExprArg(*this, ExprTys[0]));
|
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()));
|
|
|
|
|
DeclContext::lookup_const_result Lookup
|
|
|
|
|
= ClassDecl->lookup(Context, ConstructorName);
|
|
|
|
|
if (Lookup.first == Lookup.second)
|
|
|
|
|
/* No constructors */;
|
|
|
|
|
else if (OverloadedFunctionDecl *Constructors
|
|
|
|
|
= dyn_cast<OverloadedFunctionDecl>(*Lookup.first)) {
|
|
|
|
|
for (OverloadedFunctionDecl::function_iterator Con
|
|
|
|
|
= Constructors->function_begin();
|
|
|
|
|
Con != Constructors->function_end(); ++Con) {
|
|
|
|
|
CXXConstructorDecl *Constructor = cast<CXXConstructorDecl>(*Con);
|
|
|
|
|
if ((Kind == IK_Direct) ||
|
|
|
|
|
(Kind == IK_Copy && Constructor->isConvertingConstructor()) ||
|
|
|
|
|
(Kind == IK_Default && Constructor->isDefaultConstructor()))
|
|
|
|
|
AddOverloadCandidate(Constructor, Args, NumArgs, CandidateSet);
|
|
|
|
|
}
|
|
|
|
|
} else if (CXXConstructorDecl *Constructor
|
|
|
|
|
= dyn_cast<CXXConstructorDecl>(*Lookup.first)) {
|
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:
|
2008-11-22 21:44:36 +08:00
|
|
|
|
Diag(Loc, diag::err_ovl_no_viable_function_in_init)
|
|
|
|
|
<< InitEntity << (unsigned)CandidateSet.size() << Range;
|
|
|
|
|
PrintOverloadCandidates(CandidateSet, /*OnlyViable=*/false);
|
2008-11-04 04:45:27 +08:00
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
|
|
case OR_Ambiguous:
|
2008-11-20 14:06:08 +08:00
|
|
|
|
Diag(Loc, diag::err_ovl_ambiguous_init) << InitEntity << Range;
|
2008-11-04 04:45:27 +08:00
|
|
|
|
PrintOverloadCandidates(CandidateSet, /*OnlyViable=*/true);
|
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
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) {
|
2008-10-29 08:13:59 +08:00
|
|
|
|
assert(!T1->isReferenceType() && "T1 must be the pointee type of the reference type");
|
|
|
|
|
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.
|
2008-10-29 10:00:59 +08:00
|
|
|
|
bool
|
|
|
|
|
Sema::CheckReferenceInit(Expr *&Init, QualType &DeclType,
|
2008-11-04 03:09:14 +08:00
|
|
|
|
ImplicitConversionSequence *ICS,
|
|
|
|
|
bool SuppressUserConversions) {
|
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.
|
|
|
|
|
if (T2->isOverloadType()) {
|
|
|
|
|
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.
|
|
|
|
|
if (!ICS)
|
|
|
|
|
FixOverloadedFunctionReference(Init, Fn);
|
|
|
|
|
|
|
|
|
|
T2 = Fn->getType();
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2008-10-29 10:00:59 +08:00
|
|
|
|
// Compute some basic properties of the types and the initializer.
|
|
|
|
|
bool DerivedToBase = false;
|
2008-10-29 08:13:59 +08:00
|
|
|
|
Expr::isLvalueResult InitLvalue = 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
|
|
|
|
|
|
|
|
|
|
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).
|
|
|
|
|
if (InitLvalue == Expr::LV_Valid && (ICS || !Init->isBitField()) &&
|
|
|
|
|
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;
|
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.
|
2008-10-29 08:13:59 +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)),
|
2008-11-11 00:14:15 +08:00
|
|
|
|
if (!SuppressUserConversions && T2->isRecordType()) {
|
|
|
|
|
// 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);
|
|
|
|
|
|
|
|
|
|
// If the conversion function doesn't return a reference type,
|
|
|
|
|
// it can't be considered for this conversion.
|
|
|
|
|
// FIXME: This will change when we support rvalue references.
|
|
|
|
|
if (Conv->getConversionType()->isReferenceType())
|
|
|
|
|
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.
|
|
|
|
|
// 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:
|
|
|
|
|
// There was no suitable conversion; continue with other checks.
|
|
|
|
|
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
|
|
|
|
|
// type (i.e., cv1 shall be const).
|
|
|
|
|
if (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.
|
|
|
|
|
//
|
|
|
|
|
// Note that C++0x [dcl.ref.init]p5 takes away this implementation
|
|
|
|
|
// 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;
|
|
|
|
|
ICS->Standard.DirectBinding = false;
|
2008-10-29 10:00:59 +08:00
|
|
|
|
} else {
|
2008-10-29 08:13:59 +08:00
|
|
|
|
// FIXME: Binding to a subobject of the rvalue 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
|
|
|
|
}
|
|
|
|
|
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;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Actually try to convert the initializer to T1.
|
2008-10-29 10:00:59 +08:00
|
|
|
|
if (ICS) {
|
|
|
|
|
/// 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);
|
2008-10-29 10:00:59 +08:00
|
|
|
|
return ICS->ConversionKind == ImplicitConversionSequence::BadConversion;
|
|
|
|
|
} else {
|
2008-10-29 08:13:59 +08:00
|
|
|
|
return PerformImplicitConversion(Init, T1);
|
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.
|
|
|
|
|
// FIXME: Write a separate routine for checking this. For now, just
|
|
|
|
|
// allow it.
|
|
|
|
|
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-11-18 00:14:12 +08:00
|
|
|
|
if (Expr *DefArg = (*Param)->getDefaultArg())
|
|
|
|
|
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
|
|
|
|
}
|
|
|
|
|
|
2008-11-18 00:14:12 +08:00
|
|
|
|
// Overloaded operators other than operator() cannot be variadic.
|
|
|
|
|
if (Op != OO_Call &&
|
|
|
|
|
FnDecl->getType()->getAsFunctionTypeProto()->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
|
|
|
|
}
|
|
|
|
|
|
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
|
|
|
|
|
2008-12-17 15:13:27 +08:00
|
|
|
|
/// ActOnLinkageSpec - Parsed a C++ linkage-specification that
|
|
|
|
|
/// contained braces. Lang/StrSize contains the language string that
|
|
|
|
|
/// was parsed at location Loc. Decls/NumDecls provides the
|
|
|
|
|
/// declarations parsed inside the linkage specification.
|
|
|
|
|
Sema::DeclTy *Sema::ActOnLinkageSpec(SourceLocation Loc,
|
|
|
|
|
SourceLocation LBrace,
|
|
|
|
|
SourceLocation RBrace,
|
|
|
|
|
const char *Lang,
|
|
|
|
|
unsigned StrSize,
|
|
|
|
|
DeclTy **Decls, unsigned NumDecls) {
|
|
|
|
|
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 {
|
|
|
|
|
Diag(Loc, diag::err_bad_language);
|
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// FIXME: Add all the various semantics of linkage specifications
|
|
|
|
|
|
|
|
|
|
return LinkageSpecDecl::Create(Context, Loc, Language,
|
|
|
|
|
(Decl **)Decls, NumDecls);
|
|
|
|
|
}
|
|
|
|
|
|
2008-12-17 15:09:26 +08:00
|
|
|
|
Sema::DeclTy *Sema::ActOnLinkageSpec(SourceLocation Loc,
|
|
|
|
|
const char *Lang, unsigned StrSize,
|
|
|
|
|
DeclTy *D) {
|
|
|
|
|
LinkageSpecDecl::LanguageIDs Language;
|
|
|
|
|
Decl *dcl = static_cast<Decl *>(D);
|
|
|
|
|
if (strncmp(Lang, "\"C\"", StrSize) == 0)
|
|
|
|
|
Language = LinkageSpecDecl::lang_c;
|
|
|
|
|
else if (strncmp(Lang, "\"C++\"", StrSize) == 0)
|
|
|
|
|
Language = LinkageSpecDecl::lang_cxx;
|
|
|
|
|
else {
|
|
|
|
|
Diag(Loc, diag::err_bad_language);
|
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// FIXME: Add all the various semantics of linkage specifications
|
|
|
|
|
return LinkageSpecDecl::Create(Context, Loc, Language, dcl);
|
|
|
|
|
}
|
|
|
|
|
|