2008-05-02 06:18:59 +08:00
|
|
|
//===--- SemaInit.cpp - Semantic Analysis for Initializers ----------------===//
|
|
|
|
//
|
|
|
|
// The LLVM Compiler Infrastructure
|
|
|
|
//
|
|
|
|
// This file is distributed under the University of Illinois Open Source
|
|
|
|
// License. See LICENSE.TXT for details.
|
|
|
|
//
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
//
|
2011-09-25 01:47:39 +08:00
|
|
|
// This file implements semantic analysis for initializers.
|
2009-02-25 06:27:37 +08:00
|
|
|
//
|
2008-05-02 06:18:59 +08:00
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
|
2010-08-13 04:07:10 +08:00
|
|
|
#include "clang/Sema/Initialization.h"
|
2008-05-02 06:18:59 +08:00
|
|
|
#include "clang/AST/ASTContext.h"
|
2010-08-24 15:21:54 +08:00
|
|
|
#include "clang/AST/DeclObjC.h"
|
2009-05-28 00:10:08 +08:00
|
|
|
#include "clang/AST/ExprCXX.h"
|
2009-02-25 07:10:27 +08:00
|
|
|
#include "clang/AST/ExprObjC.h"
|
2009-12-22 23:35:07 +08:00
|
|
|
#include "clang/AST/TypeLoc.h"
|
2012-12-04 17:13:33 +08:00
|
|
|
#include "clang/Lex/Preprocessor.h"
|
|
|
|
#include "clang/Sema/Designator.h"
|
|
|
|
#include "clang/Sema/Lookup.h"
|
|
|
|
#include "clang/Sema/SemaInternal.h"
|
2012-01-18 06:49:42 +08:00
|
|
|
#include "llvm/ADT/APInt.h"
|
2012-02-04 21:45:25 +08:00
|
|
|
#include "llvm/ADT/SmallString.h"
|
2009-12-10 07:02:17 +08:00
|
|
|
#include "llvm/Support/ErrorHandling.h"
|
2011-07-27 07:20:30 +08:00
|
|
|
#include "llvm/Support/raw_ostream.h"
|
2009-01-29 08:45:39 +08:00
|
|
|
#include <map>
|
2009-01-22 08:58:24 +08:00
|
|
|
using namespace clang;
|
2008-05-02 06:18:59 +08:00
|
|
|
|
2009-02-25 06:27:37 +08:00
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
// Sema Initialization Checking
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
|
2013-05-15 19:03:04 +08:00
|
|
|
/// \brief Check whether T is compatible with a wide character type (wchar_t,
|
|
|
|
/// char16_t or char32_t).
|
|
|
|
static bool IsWideCharCompatible(QualType T, ASTContext &Context) {
|
|
|
|
if (Context.typesAreCompatible(Context.getWideCharType(), T))
|
|
|
|
return true;
|
|
|
|
if (Context.getLangOpts().CPlusPlus || Context.getLangOpts().C11) {
|
|
|
|
return Context.typesAreCompatible(Context.Char16Ty, T) ||
|
|
|
|
Context.typesAreCompatible(Context.Char32Ty, T);
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
enum StringInitFailureKind {
|
|
|
|
SIF_None,
|
|
|
|
SIF_NarrowStringIntoWideChar,
|
|
|
|
SIF_WideStringIntoChar,
|
|
|
|
SIF_IncompatWideStringIntoWideChar,
|
|
|
|
SIF_Other
|
|
|
|
};
|
|
|
|
|
|
|
|
/// \brief Check whether the array of type AT can be initialized by the Init
|
|
|
|
/// expression by means of string initialization. Returns SIF_None if so,
|
|
|
|
/// otherwise returns a StringInitFailureKind that describes why the
|
|
|
|
/// initialization would not work.
|
|
|
|
static StringInitFailureKind IsStringInit(Expr *Init, const ArrayType *AT,
|
|
|
|
ASTContext &Context) {
|
2009-05-30 02:22:49 +08:00
|
|
|
if (!isa<ConstantArrayType>(AT) && !isa<IncompleteArrayType>(AT))
|
2013-05-15 19:03:04 +08:00
|
|
|
return SIF_Other;
|
2009-05-30 02:22:49 +08:00
|
|
|
|
2009-02-27 07:26:43 +08:00
|
|
|
// See if this is a string literal or @encode.
|
|
|
|
Init = Init->IgnoreParens();
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-02-27 07:26:43 +08:00
|
|
|
// Handle @encode, which is a narrow string.
|
|
|
|
if (isa<ObjCEncodeExpr>(Init) && AT->getElementType()->isCharType())
|
2013-05-15 19:03:04 +08:00
|
|
|
return SIF_None;
|
2009-02-27 07:26:43 +08:00
|
|
|
|
|
|
|
// Otherwise we can only handle string literals.
|
|
|
|
StringLiteral *SL = dyn_cast<StringLiteral>(Init);
|
2013-05-15 19:03:04 +08:00
|
|
|
if (SL == 0)
|
|
|
|
return SIF_Other;
|
2009-05-31 18:54:53 +08:00
|
|
|
|
2013-05-15 19:03:04 +08:00
|
|
|
const QualType ElemTy =
|
|
|
|
Context.getCanonicalType(AT->getElementType()).getUnqualifiedType();
|
2011-07-27 13:40:30 +08:00
|
|
|
|
|
|
|
switch (SL->getKind()) {
|
|
|
|
case StringLiteral::Ascii:
|
|
|
|
case StringLiteral::UTF8:
|
|
|
|
// char array can be initialized with a narrow string.
|
|
|
|
// Only allow char x[] = "foo"; not char x[] = L"foo";
|
2013-05-15 19:03:04 +08:00
|
|
|
if (ElemTy->isCharType())
|
|
|
|
return SIF_None;
|
|
|
|
if (IsWideCharCompatible(ElemTy, Context))
|
|
|
|
return SIF_NarrowStringIntoWideChar;
|
|
|
|
return SIF_Other;
|
|
|
|
// C99 6.7.8p15 (with correction from DR343), or C11 6.7.9p15:
|
|
|
|
// "An array with element type compatible with a qualified or unqualified
|
|
|
|
// version of wchar_t, char16_t, or char32_t may be initialized by a wide
|
|
|
|
// string literal with the corresponding encoding prefix (L, u, or U,
|
|
|
|
// respectively), optionally enclosed in braces.
|
2011-07-27 13:40:30 +08:00
|
|
|
case StringLiteral::UTF16:
|
2013-05-15 19:03:04 +08:00
|
|
|
if (Context.typesAreCompatible(Context.Char16Ty, ElemTy))
|
|
|
|
return SIF_None;
|
|
|
|
if (ElemTy->isCharType())
|
|
|
|
return SIF_WideStringIntoChar;
|
|
|
|
if (IsWideCharCompatible(ElemTy, Context))
|
|
|
|
return SIF_IncompatWideStringIntoWideChar;
|
|
|
|
return SIF_Other;
|
2011-07-27 13:40:30 +08:00
|
|
|
case StringLiteral::UTF32:
|
2013-05-15 19:03:04 +08:00
|
|
|
if (Context.typesAreCompatible(Context.Char32Ty, ElemTy))
|
|
|
|
return SIF_None;
|
|
|
|
if (ElemTy->isCharType())
|
|
|
|
return SIF_WideStringIntoChar;
|
|
|
|
if (IsWideCharCompatible(ElemTy, Context))
|
|
|
|
return SIF_IncompatWideStringIntoWideChar;
|
|
|
|
return SIF_Other;
|
2011-07-27 13:40:30 +08:00
|
|
|
case StringLiteral::Wide:
|
2013-05-15 19:03:04 +08:00
|
|
|
if (Context.typesAreCompatible(Context.getWideCharType(), ElemTy))
|
|
|
|
return SIF_None;
|
|
|
|
if (ElemTy->isCharType())
|
|
|
|
return SIF_WideStringIntoChar;
|
|
|
|
if (IsWideCharCompatible(ElemTy, Context))
|
|
|
|
return SIF_IncompatWideStringIntoWideChar;
|
|
|
|
return SIF_Other;
|
2011-07-27 13:40:30 +08:00
|
|
|
}
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2011-07-27 13:40:30 +08:00
|
|
|
llvm_unreachable("missed a StringLiteral kind?");
|
2009-02-25 06:27:37 +08:00
|
|
|
}
|
|
|
|
|
2013-05-16 17:22:40 +08:00
|
|
|
static StringInitFailureKind IsStringInit(Expr *init, QualType declType,
|
|
|
|
ASTContext &Context) {
|
2011-02-21 15:22:22 +08:00
|
|
|
const ArrayType *arrayType = Context.getAsArrayType(declType);
|
2013-05-15 19:03:04 +08:00
|
|
|
if (!arrayType)
|
2013-05-16 17:22:40 +08:00
|
|
|
return SIF_Other;
|
|
|
|
return IsStringInit(init, arrayType, Context);
|
2011-02-21 15:22:22 +08:00
|
|
|
}
|
|
|
|
|
2013-05-06 00:40:13 +08:00
|
|
|
/// Update the type of a string literal, including any surrounding parentheses,
|
|
|
|
/// to match the type of the object which it is initializing.
|
|
|
|
static void updateStringLiteralType(Expr *E, QualType Ty) {
|
2013-05-06 08:35:47 +08:00
|
|
|
while (true) {
|
2013-05-06 00:40:13 +08:00
|
|
|
E->setType(Ty);
|
2013-05-06 08:35:47 +08:00
|
|
|
if (isa<StringLiteral>(E) || isa<ObjCEncodeExpr>(E))
|
|
|
|
break;
|
|
|
|
else if (ParenExpr *PE = dyn_cast<ParenExpr>(E))
|
|
|
|
E = PE->getSubExpr();
|
|
|
|
else if (UnaryOperator *UO = dyn_cast<UnaryOperator>(E))
|
|
|
|
E = UO->getSubExpr();
|
|
|
|
else if (GenericSelectionExpr *GSE = dyn_cast<GenericSelectionExpr>(E))
|
|
|
|
E = GSE->getResultExpr();
|
|
|
|
else
|
|
|
|
llvm_unreachable("unexpected expr in string literal init");
|
2013-05-06 00:40:13 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-02-21 15:57:55 +08:00
|
|
|
static void CheckStringInit(Expr *Str, QualType &DeclT, const ArrayType *AT,
|
|
|
|
Sema &S) {
|
2009-02-25 07:10:27 +08:00
|
|
|
// Get the length of the string as parsed.
|
|
|
|
uint64_t StrLength =
|
|
|
|
cast<ConstantArrayType>(Str->getType())->getSize().getZExtValue();
|
|
|
|
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-02-25 06:27:37 +08:00
|
|
|
if (const IncompleteArrayType *IAT = dyn_cast<IncompleteArrayType>(AT)) {
|
2009-09-09 23:08:12 +08:00
|
|
|
// C99 6.7.8p14. We have an array of character type with unknown size
|
2009-02-25 06:27:37 +08:00
|
|
|
// being initialized to a string literal.
|
2012-08-05 01:00:46 +08:00
|
|
|
llvm::APInt ConstVal(32, StrLength);
|
2009-02-25 06:27:37 +08:00
|
|
|
// Return a new array type (C99 6.7.8p22).
|
2009-10-16 08:14:28 +08:00
|
|
|
DeclT = S.Context.getConstantArrayType(IAT->getElementType(),
|
|
|
|
ConstVal,
|
|
|
|
ArrayType::Normal, 0);
|
2013-05-06 00:40:13 +08:00
|
|
|
updateStringLiteralType(Str, DeclT);
|
2009-02-25 07:01:39 +08:00
|
|
|
return;
|
2009-02-25 06:27:37 +08:00
|
|
|
}
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-05-30 02:22:49 +08:00
|
|
|
const ConstantArrayType *CAT = cast<ConstantArrayType>(AT);
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2011-04-11 08:23:45 +08:00
|
|
|
// We have an array of character type with known size. However,
|
2009-05-30 02:22:49 +08:00
|
|
|
// the size may be smaller or larger than the string we are initializing.
|
|
|
|
// FIXME: Avoid truncation for 64-bit length strings.
|
2012-03-11 15:00:24 +08:00
|
|
|
if (S.getLangOpts().CPlusPlus) {
|
2013-05-06 00:40:13 +08:00
|
|
|
if (StringLiteral *SL = dyn_cast<StringLiteral>(Str->IgnoreParens())) {
|
2011-04-14 08:41:11 +08:00
|
|
|
// For Pascal strings it's OK to strip off the terminating null character,
|
|
|
|
// so the example below is valid:
|
|
|
|
//
|
|
|
|
// unsigned char a[2] = "\pa";
|
|
|
|
if (SL->isPascal())
|
|
|
|
StrLength--;
|
|
|
|
}
|
|
|
|
|
2011-04-11 08:23:45 +08:00
|
|
|
// [dcl.init.string]p2
|
|
|
|
if (StrLength > CAT->getSize().getZExtValue())
|
2012-03-10 02:35:03 +08:00
|
|
|
S.Diag(Str->getLocStart(),
|
2011-04-11 08:23:45 +08:00
|
|
|
diag::err_initializer_string_for_char_array_too_long)
|
|
|
|
<< Str->getSourceRange();
|
|
|
|
} else {
|
|
|
|
// C99 6.7.8p14.
|
|
|
|
if (StrLength-1 > CAT->getSize().getZExtValue())
|
2012-03-10 02:35:03 +08:00
|
|
|
S.Diag(Str->getLocStart(),
|
2011-04-11 08:23:45 +08:00
|
|
|
diag::warn_initializer_string_for_char_array_too_long)
|
|
|
|
<< Str->getSourceRange();
|
|
|
|
}
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-05-30 02:22:49 +08:00
|
|
|
// Set the type to the actual size that we are initializing. If we have
|
|
|
|
// something like:
|
|
|
|
// char x[1] = "foo";
|
|
|
|
// then this will set the string literal's type to char[1].
|
2013-05-06 00:40:13 +08:00
|
|
|
updateStringLiteralType(Str, DeclT);
|
2009-02-25 06:27:37 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
// Semantic checking for initializer lists.
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
|
2009-01-29 09:05:33 +08:00
|
|
|
/// @brief Semantic checking for initializer lists.
|
|
|
|
///
|
|
|
|
/// The InitListChecker class contains a set of routines that each
|
|
|
|
/// handle the initialization of a certain kind of entity, e.g.,
|
|
|
|
/// arrays, vectors, struct/union types, scalars, etc. The
|
|
|
|
/// InitListChecker itself performs a recursive walk of the subobject
|
|
|
|
/// structure of the type to be initialized, while stepping through
|
|
|
|
/// the initializer list one element at a time. The IList and Index
|
|
|
|
/// parameters to each of the Check* routines contain the active
|
|
|
|
/// (syntactic) initializer list and the index into that initializer
|
|
|
|
/// list that represents the current initializer. Each routine is
|
|
|
|
/// responsible for moving that Index forward as it consumes elements.
|
|
|
|
///
|
|
|
|
/// Each Check* routine also has a StructuredList/StructuredIndex
|
2011-01-28 03:55:10 +08:00
|
|
|
/// arguments, which contains the current "structured" (semantic)
|
2009-01-29 09:05:33 +08:00
|
|
|
/// initializer list and the index into that initializer list where we
|
|
|
|
/// are copying initializers as we map them over to the semantic
|
|
|
|
/// list. Once we have completed our recursive walk of the subobject
|
|
|
|
/// structure, we will have constructed a full semantic initializer
|
|
|
|
/// list.
|
|
|
|
///
|
|
|
|
/// C99 designators cause changes in the initializer list traversal,
|
|
|
|
/// because they make the initialization "jump" into a specific
|
|
|
|
/// subobject and then continue the initialization from that
|
|
|
|
/// point. CheckDesignatedInitializer() recursively steps into the
|
|
|
|
/// designated subobject and manages backing out the recursion to
|
|
|
|
/// initialize the subobjects after the one designated.
|
2009-02-25 06:48:58 +08:00
|
|
|
namespace {
|
2009-01-29 08:45:39 +08:00
|
|
|
class InitListChecker {
|
2009-02-25 06:50:46 +08:00
|
|
|
Sema &SemaRef;
|
2009-01-29 08:45:39 +08:00
|
|
|
bool hadError;
|
2011-09-25 01:48:00 +08:00
|
|
|
bool VerifyOnly; // no diagnostics, no structure building
|
2012-02-23 22:48:40 +08:00
|
|
|
llvm::DenseMap<InitListExpr *, InitListExpr *> SyntacticToSemantic;
|
2009-01-29 08:45:39 +08:00
|
|
|
InitListExpr *FullyStructuredList;
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2010-01-24 07:23:01 +08:00
|
|
|
void CheckImplicitInitList(const InitializedEntity &Entity,
|
2010-01-24 04:47:59 +08:00
|
|
|
InitListExpr *ParentIList, QualType T,
|
2009-01-29 09:05:33 +08:00
|
|
|
unsigned &Index, InitListExpr *StructuredList,
|
2011-08-24 04:17:13 +08:00
|
|
|
unsigned &StructuredIndex);
|
2010-01-24 07:23:01 +08:00
|
|
|
void CheckExplicitInitList(const InitializedEntity &Entity,
|
2010-01-24 03:55:29 +08:00
|
|
|
InitListExpr *IList, QualType &T,
|
2013-09-21 04:10:22 +08:00
|
|
|
InitListExpr *StructuredList,
|
2009-02-05 06:46:25 +08:00
|
|
|
bool TopLevelObject = false);
|
2010-01-24 07:23:01 +08:00
|
|
|
void CheckListElementTypes(const InitializedEntity &Entity,
|
2010-01-24 03:55:29 +08:00
|
|
|
InitListExpr *IList, QualType &DeclType,
|
2009-09-09 23:08:12 +08:00
|
|
|
bool SubobjectIsDesignatorContext,
|
2009-01-29 08:45:39 +08:00
|
|
|
unsigned &Index,
|
2009-01-29 09:05:33 +08:00
|
|
|
InitListExpr *StructuredList,
|
2009-02-05 06:46:25 +08:00
|
|
|
unsigned &StructuredIndex,
|
|
|
|
bool TopLevelObject = false);
|
2010-01-24 07:23:01 +08:00
|
|
|
void CheckSubElementType(const InitializedEntity &Entity,
|
2010-01-24 03:55:29 +08:00
|
|
|
InitListExpr *IList, QualType ElemType,
|
2009-01-29 08:45:39 +08:00
|
|
|
unsigned &Index,
|
2009-01-29 09:05:33 +08:00
|
|
|
InitListExpr *StructuredList,
|
|
|
|
unsigned &StructuredIndex);
|
2011-09-20 07:17:44 +08:00
|
|
|
void CheckComplexType(const InitializedEntity &Entity,
|
|
|
|
InitListExpr *IList, QualType DeclType,
|
|
|
|
unsigned &Index,
|
|
|
|
InitListExpr *StructuredList,
|
|
|
|
unsigned &StructuredIndex);
|
2010-01-24 07:23:01 +08:00
|
|
|
void CheckScalarType(const InitializedEntity &Entity,
|
2010-01-24 03:55:29 +08:00
|
|
|
InitListExpr *IList, QualType DeclType,
|
2009-01-29 08:45:39 +08:00
|
|
|
unsigned &Index,
|
2009-01-29 09:05:33 +08:00
|
|
|
InitListExpr *StructuredList,
|
|
|
|
unsigned &StructuredIndex);
|
2010-01-24 07:23:01 +08:00
|
|
|
void CheckReferenceType(const InitializedEntity &Entity,
|
|
|
|
InitListExpr *IList, QualType DeclType,
|
2009-01-31 06:09:00 +08:00
|
|
|
unsigned &Index,
|
|
|
|
InitListExpr *StructuredList,
|
|
|
|
unsigned &StructuredIndex);
|
2010-01-24 07:23:01 +08:00
|
|
|
void CheckVectorType(const InitializedEntity &Entity,
|
2010-01-24 03:55:29 +08:00
|
|
|
InitListExpr *IList, QualType DeclType, unsigned &Index,
|
2009-01-29 09:05:33 +08:00
|
|
|
InitListExpr *StructuredList,
|
|
|
|
unsigned &StructuredIndex);
|
2010-01-24 07:23:01 +08:00
|
|
|
void CheckStructUnionTypes(const InitializedEntity &Entity,
|
2010-01-24 04:20:40 +08:00
|
|
|
InitListExpr *IList, QualType DeclType,
|
2009-09-09 23:08:12 +08:00
|
|
|
RecordDecl::field_iterator Field,
|
2009-01-29 08:45:39 +08:00
|
|
|
bool SubobjectIsDesignatorContext, unsigned &Index,
|
2009-01-29 09:05:33 +08:00
|
|
|
InitListExpr *StructuredList,
|
2009-02-05 06:46:25 +08:00
|
|
|
unsigned &StructuredIndex,
|
|
|
|
bool TopLevelObject = false);
|
2010-01-24 07:23:01 +08:00
|
|
|
void CheckArrayType(const InitializedEntity &Entity,
|
2010-01-24 04:13:41 +08:00
|
|
|
InitListExpr *IList, QualType &DeclType,
|
2009-09-09 23:08:12 +08:00
|
|
|
llvm::APSInt elementIndex,
|
2009-01-29 08:45:39 +08:00
|
|
|
bool SubobjectIsDesignatorContext, unsigned &Index,
|
2009-01-29 09:05:33 +08:00
|
|
|
InitListExpr *StructuredList,
|
|
|
|
unsigned &StructuredIndex);
|
2010-01-24 07:23:01 +08:00
|
|
|
bool CheckDesignatedInitializer(const InitializedEntity &Entity,
|
2010-01-24 06:49:02 +08:00
|
|
|
InitListExpr *IList, DesignatedInitExpr *DIE,
|
2009-04-15 12:56:10 +08:00
|
|
|
unsigned DesigIdx,
|
2009-09-09 23:08:12 +08:00
|
|
|
QualType &CurrentObjectType,
|
2009-01-29 08:45:39 +08:00
|
|
|
RecordDecl::field_iterator *NextField,
|
|
|
|
llvm::APSInt *NextElementIndex,
|
|
|
|
unsigned &Index,
|
|
|
|
InitListExpr *StructuredList,
|
|
|
|
unsigned &StructuredIndex,
|
2009-02-05 06:46:25 +08:00
|
|
|
bool FinishSubobjectInit,
|
|
|
|
bool TopLevelObject);
|
2009-01-29 08:45:39 +08:00
|
|
|
InitListExpr *getStructuredSubobjectInit(InitListExpr *IList, unsigned Index,
|
|
|
|
QualType CurrentObjectType,
|
|
|
|
InitListExpr *StructuredList,
|
|
|
|
unsigned StructuredIndex,
|
|
|
|
SourceRange InitRange);
|
2009-01-29 09:05:33 +08:00
|
|
|
void UpdateStructuredListElement(InitListExpr *StructuredList,
|
|
|
|
unsigned &StructuredIndex,
|
2009-01-29 08:45:39 +08:00
|
|
|
Expr *expr);
|
|
|
|
int numArrayElements(QualType DeclType);
|
|
|
|
int numStructUnionElements(QualType DeclType);
|
2009-01-31 06:09:00 +08:00
|
|
|
|
2009-12-22 08:05:34 +08:00
|
|
|
void FillInValueInitForField(unsigned Init, FieldDecl *Field,
|
|
|
|
const InitializedEntity &ParentEntity,
|
|
|
|
InitListExpr *ILE, bool &RequiresSecondPass);
|
2009-12-16 14:35:08 +08:00
|
|
|
void FillInValueInitializations(const InitializedEntity &Entity,
|
|
|
|
InitListExpr *ILE, bool &RequiresSecondPass);
|
2011-08-24 06:24:57 +08:00
|
|
|
bool CheckFlexibleArrayInit(const InitializedEntity &Entity,
|
|
|
|
Expr *InitExpr, FieldDecl *Field,
|
|
|
|
bool TopLevelObject);
|
2011-10-17 02:19:20 +08:00
|
|
|
void CheckValueInitializable(const InitializedEntity &Entity);
|
|
|
|
|
2009-01-29 08:45:39 +08:00
|
|
|
public:
|
2009-12-16 14:35:08 +08:00
|
|
|
InitListChecker(Sema &S, const InitializedEntity &Entity,
|
2013-06-06 19:41:05 +08:00
|
|
|
InitListExpr *IL, QualType &T, bool VerifyOnly);
|
2009-01-29 08:45:39 +08:00
|
|
|
bool HadError() { return hadError; }
|
|
|
|
|
|
|
|
// @brief Retrieves the fully-structured initializer list used for
|
|
|
|
// semantic analysis and code generation.
|
|
|
|
InitListExpr *getFullyStructuredList() const { return FullyStructuredList; }
|
|
|
|
};
|
2009-02-25 06:48:58 +08:00
|
|
|
} // end anonymous namespace
|
2009-01-29 13:10:57 +08:00
|
|
|
|
2011-10-17 02:19:20 +08:00
|
|
|
void InitListChecker::CheckValueInitializable(const InitializedEntity &Entity) {
|
|
|
|
assert(VerifyOnly &&
|
|
|
|
"CheckValueInitializable is only inteded for verification mode.");
|
|
|
|
|
|
|
|
SourceLocation Loc;
|
|
|
|
InitializationKind Kind = InitializationKind::CreateValue(Loc, Loc, Loc,
|
|
|
|
true);
|
2013-05-06 04:40:26 +08:00
|
|
|
InitializationSequence InitSeq(SemaRef, Entity, Kind, None);
|
2011-10-17 02:19:20 +08:00
|
|
|
if (InitSeq.Failed())
|
|
|
|
hadError = true;
|
|
|
|
}
|
|
|
|
|
2009-12-22 08:05:34 +08:00
|
|
|
void InitListChecker::FillInValueInitForField(unsigned Init, FieldDecl *Field,
|
|
|
|
const InitializedEntity &ParentEntity,
|
2011-01-27 15:10:08 +08:00
|
|
|
InitListExpr *ILE,
|
2009-12-22 08:05:34 +08:00
|
|
|
bool &RequiresSecondPass) {
|
2012-03-10 02:35:03 +08:00
|
|
|
SourceLocation Loc = ILE->getLocStart();
|
2009-12-22 08:05:34 +08:00
|
|
|
unsigned NumInits = ILE->getNumInits();
|
2011-01-27 15:10:08 +08:00
|
|
|
InitializedEntity MemberEntity
|
2009-12-22 08:05:34 +08:00
|
|
|
= InitializedEntity::InitializeMember(Field, &ParentEntity);
|
|
|
|
if (Init >= NumInits || !ILE->getInit(Init)) {
|
2013-04-21 06:23:05 +08:00
|
|
|
// If there's no explicit initializer but we have a default initializer, use
|
|
|
|
// that. This only happens in C++1y, since classes with default
|
|
|
|
// initializers are not aggregates in C++11.
|
|
|
|
if (Field->hasInClassInitializer()) {
|
|
|
|
Expr *DIE = CXXDefaultInitExpr::Create(SemaRef.Context,
|
|
|
|
ILE->getRBraceLoc(), Field);
|
|
|
|
if (Init < NumInits)
|
|
|
|
ILE->setInit(Init, DIE);
|
|
|
|
else {
|
|
|
|
ILE->updateInit(SemaRef.Context, Init, DIE);
|
|
|
|
RequiresSecondPass = true;
|
|
|
|
}
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2009-12-22 08:05:34 +08:00
|
|
|
// FIXME: We probably don't need to handle references
|
|
|
|
// specially here, since value-initialization of references is
|
|
|
|
// handled in InitializationSequence.
|
|
|
|
if (Field->getType()->isReferenceType()) {
|
|
|
|
// C++ [dcl.init.aggr]p9:
|
|
|
|
// If an incomplete or empty initializer-list leaves a
|
|
|
|
// member of reference type uninitialized, the program is
|
|
|
|
// ill-formed.
|
|
|
|
SemaRef.Diag(Loc, diag::err_init_reference_member_uninitialized)
|
|
|
|
<< Field->getType()
|
|
|
|
<< ILE->getSyntacticForm()->getSourceRange();
|
|
|
|
SemaRef.Diag(Field->getLocation(),
|
|
|
|
diag::note_uninit_reference_member);
|
|
|
|
hadError = true;
|
|
|
|
return;
|
|
|
|
}
|
2011-01-27 15:10:08 +08:00
|
|
|
|
2009-12-22 08:05:34 +08:00
|
|
|
InitializationKind Kind = InitializationKind::CreateValue(Loc, Loc, Loc,
|
|
|
|
true);
|
2013-05-06 04:40:26 +08:00
|
|
|
InitializationSequence InitSeq(SemaRef, MemberEntity, Kind, None);
|
2009-12-22 08:05:34 +08:00
|
|
|
if (!InitSeq) {
|
2013-05-05 08:41:58 +08:00
|
|
|
InitSeq.Diagnose(SemaRef, MemberEntity, Kind, None);
|
2009-12-22 08:05:34 +08:00
|
|
|
hadError = true;
|
|
|
|
return;
|
|
|
|
}
|
2011-01-27 15:10:08 +08:00
|
|
|
|
2010-08-24 14:29:42 +08:00
|
|
|
ExprResult MemberInit
|
2013-05-06 04:40:26 +08:00
|
|
|
= InitSeq.Perform(SemaRef, MemberEntity, Kind, None);
|
2009-12-22 08:05:34 +08:00
|
|
|
if (MemberInit.isInvalid()) {
|
|
|
|
hadError = true;
|
|
|
|
return;
|
|
|
|
}
|
2011-01-27 15:10:08 +08:00
|
|
|
|
2009-12-22 08:05:34 +08:00
|
|
|
if (hadError) {
|
|
|
|
// Do nothing
|
|
|
|
} else if (Init < NumInits) {
|
|
|
|
ILE->setInit(Init, MemberInit.takeAs<Expr>());
|
2011-06-05 21:59:11 +08:00
|
|
|
} else if (InitSeq.isConstructorInitialization()) {
|
2009-12-22 08:05:34 +08:00
|
|
|
// Value-initialization requires a constructor call, so
|
|
|
|
// extend the initializer list to include the constructor
|
|
|
|
// call and make a note that we'll need to take another pass
|
|
|
|
// through the initializer list.
|
2010-04-14 07:39:13 +08:00
|
|
|
ILE->updateInit(SemaRef.Context, Init, MemberInit.takeAs<Expr>());
|
2009-12-22 08:05:34 +08:00
|
|
|
RequiresSecondPass = true;
|
|
|
|
}
|
|
|
|
} else if (InitListExpr *InnerILE
|
|
|
|
= dyn_cast<InitListExpr>(ILE->getInit(Init)))
|
2011-01-27 15:10:08 +08:00
|
|
|
FillInValueInitializations(MemberEntity, InnerILE,
|
|
|
|
RequiresSecondPass);
|
2009-12-22 08:05:34 +08:00
|
|
|
}
|
|
|
|
|
2009-01-29 05:54:33 +08:00
|
|
|
/// Recursively replaces NULL values within the given initializer list
|
|
|
|
/// with expressions that perform value-initialization of the
|
|
|
|
/// appropriate type.
|
2011-01-27 15:10:08 +08:00
|
|
|
void
|
2009-12-16 14:35:08 +08:00
|
|
|
InitListChecker::FillInValueInitializations(const InitializedEntity &Entity,
|
|
|
|
InitListExpr *ILE,
|
|
|
|
bool &RequiresSecondPass) {
|
2009-09-09 23:08:12 +08:00
|
|
|
assert((ILE->getType() != SemaRef.Context.VoidTy) &&
|
2009-01-31 06:09:00 +08:00
|
|
|
"Should not have void type");
|
2012-03-10 02:35:03 +08:00
|
|
|
SourceLocation Loc = ILE->getLocStart();
|
2009-02-03 01:43:21 +08:00
|
|
|
if (ILE->getSyntacticForm())
|
2012-03-10 02:35:03 +08:00
|
|
|
Loc = ILE->getSyntacticForm()->getLocStart();
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-07-30 05:53:49 +08:00
|
|
|
if (const RecordType *RType = ILE->getType()->getAs<RecordType>()) {
|
2013-04-21 06:23:05 +08:00
|
|
|
const RecordDecl *RDecl = RType->getDecl();
|
|
|
|
if (RDecl->isUnion() && ILE->getInitializedFieldInUnion())
|
2009-12-22 08:05:34 +08:00
|
|
|
FillInValueInitForField(0, ILE->getInitializedFieldInUnion(),
|
|
|
|
Entity, ILE, RequiresSecondPass);
|
2013-04-21 06:23:05 +08:00
|
|
|
else if (RDecl->isUnion() && isa<CXXRecordDecl>(RDecl) &&
|
|
|
|
cast<CXXRecordDecl>(RDecl)->hasInClassInitializer()) {
|
|
|
|
for (RecordDecl::field_iterator Field = RDecl->field_begin(),
|
|
|
|
FieldEnd = RDecl->field_end();
|
|
|
|
Field != FieldEnd; ++Field) {
|
|
|
|
if (Field->hasInClassInitializer()) {
|
|
|
|
FillInValueInitForField(0, *Field, Entity, ILE, RequiresSecondPass);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
} else {
|
2009-12-22 08:05:34 +08:00
|
|
|
unsigned Init = 0;
|
2013-04-21 06:23:05 +08:00
|
|
|
for (RecordDecl::field_iterator Field = RDecl->field_begin(),
|
|
|
|
FieldEnd = RDecl->field_end();
|
2009-12-22 08:05:34 +08:00
|
|
|
Field != FieldEnd; ++Field) {
|
|
|
|
if (Field->isUnnamedBitfield())
|
|
|
|
continue;
|
|
|
|
|
|
|
|
if (hadError)
|
2009-02-03 01:43:21 +08:00
|
|
|
return;
|
|
|
|
|
2012-06-07 04:45:41 +08:00
|
|
|
FillInValueInitForField(Init, *Field, Entity, ILE, RequiresSecondPass);
|
2009-12-22 08:05:34 +08:00
|
|
|
if (hadError)
|
2009-12-16 14:35:08 +08:00
|
|
|
return;
|
|
|
|
|
2009-12-22 08:05:34 +08:00
|
|
|
++Init;
|
|
|
|
|
|
|
|
// Only look at the first initialization of a union.
|
2013-04-21 06:23:05 +08:00
|
|
|
if (RDecl->isUnion())
|
2009-12-22 08:05:34 +08:00
|
|
|
break;
|
|
|
|
}
|
2009-01-29 05:54:33 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
return;
|
2009-09-09 23:08:12 +08:00
|
|
|
}
|
2009-01-29 05:54:33 +08:00
|
|
|
|
|
|
|
QualType ElementType;
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-12-16 14:35:08 +08:00
|
|
|
InitializedEntity ElementEntity = Entity;
|
2009-02-03 01:43:21 +08:00
|
|
|
unsigned NumInits = ILE->getNumInits();
|
|
|
|
unsigned NumElements = NumInits;
|
2009-02-25 06:50:46 +08:00
|
|
|
if (const ArrayType *AType = SemaRef.Context.getAsArrayType(ILE->getType())) {
|
2009-01-29 05:54:33 +08:00
|
|
|
ElementType = AType->getElementType();
|
2009-02-03 01:43:21 +08:00
|
|
|
if (const ConstantArrayType *CAType = dyn_cast<ConstantArrayType>(AType))
|
|
|
|
NumElements = CAType->getSize().getZExtValue();
|
2011-01-27 15:10:08 +08:00
|
|
|
ElementEntity = InitializedEntity::InitializeElement(SemaRef.Context,
|
2009-12-16 14:35:08 +08:00
|
|
|
0, Entity);
|
2009-09-22 07:43:11 +08:00
|
|
|
} else if (const VectorType *VType = ILE->getType()->getAs<VectorType>()) {
|
2009-01-29 05:54:33 +08:00
|
|
|
ElementType = VType->getElementType();
|
2009-02-03 01:43:21 +08:00
|
|
|
NumElements = VType->getNumElements();
|
2011-01-27 15:10:08 +08:00
|
|
|
ElementEntity = InitializedEntity::InitializeElement(SemaRef.Context,
|
2009-12-16 14:35:08 +08:00
|
|
|
0, Entity);
|
2009-09-09 23:08:12 +08:00
|
|
|
} else
|
2009-01-29 05:54:33 +08:00
|
|
|
ElementType = ILE->getType();
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2011-01-27 15:10:08 +08:00
|
|
|
|
2009-02-03 01:43:21 +08:00
|
|
|
for (unsigned Init = 0; Init != NumElements; ++Init) {
|
2009-12-17 02:50:27 +08:00
|
|
|
if (hadError)
|
|
|
|
return;
|
|
|
|
|
2010-01-23 12:34:47 +08:00
|
|
|
if (ElementEntity.getKind() == InitializedEntity::EK_ArrayElement ||
|
|
|
|
ElementEntity.getKind() == InitializedEntity::EK_VectorElement)
|
2009-12-16 14:35:08 +08:00
|
|
|
ElementEntity.setElementIndex(Init);
|
|
|
|
|
2011-10-22 07:02:22 +08:00
|
|
|
Expr *InitExpr = (Init < NumInits ? ILE->getInit(Init) : 0);
|
|
|
|
if (!InitExpr && !ILE->hasArrayFiller()) {
|
2009-12-16 14:35:08 +08:00
|
|
|
InitializationKind Kind = InitializationKind::CreateValue(Loc, Loc, Loc,
|
|
|
|
true);
|
2013-05-06 04:40:26 +08:00
|
|
|
InitializationSequence InitSeq(SemaRef, ElementEntity, Kind, None);
|
2009-12-16 14:35:08 +08:00
|
|
|
if (!InitSeq) {
|
2013-05-05 08:41:58 +08:00
|
|
|
InitSeq.Diagnose(SemaRef, ElementEntity, Kind, None);
|
2009-02-03 01:43:21 +08:00
|
|
|
hadError = true;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2010-08-24 14:29:42 +08:00
|
|
|
ExprResult ElementInit
|
2013-05-06 04:40:26 +08:00
|
|
|
= InitSeq.Perform(SemaRef, ElementEntity, Kind, None);
|
2009-12-16 14:35:08 +08:00
|
|
|
if (ElementInit.isInvalid()) {
|
2009-12-17 02:50:27 +08:00
|
|
|
hadError = true;
|
2009-12-16 14:35:08 +08:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (hadError) {
|
|
|
|
// Do nothing
|
|
|
|
} else if (Init < NumInits) {
|
2011-04-22 04:03:38 +08:00
|
|
|
// For arrays, just set the expression used for value-initialization
|
|
|
|
// of the "holes" in the array.
|
|
|
|
if (ElementEntity.getKind() == InitializedEntity::EK_ArrayElement)
|
|
|
|
ILE->setArrayFiller(ElementInit.takeAs<Expr>());
|
|
|
|
else
|
|
|
|
ILE->setInit(Init, ElementInit.takeAs<Expr>());
|
2011-04-21 08:27:41 +08:00
|
|
|
} else {
|
|
|
|
// For arrays, just set the expression used for value-initialization
|
|
|
|
// of the rest of elements and exit.
|
|
|
|
if (ElementEntity.getKind() == InitializedEntity::EK_ArrayElement) {
|
|
|
|
ILE->setArrayFiller(ElementInit.takeAs<Expr>());
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2011-06-05 21:59:11 +08:00
|
|
|
if (InitSeq.isConstructorInitialization()) {
|
2011-04-21 08:27:41 +08:00
|
|
|
// Value-initialization requires a constructor call, so
|
|
|
|
// extend the initializer list to include the constructor
|
|
|
|
// call and make a note that we'll need to take another pass
|
|
|
|
// through the initializer list.
|
|
|
|
ILE->updateInit(SemaRef.Context, Init, ElementInit.takeAs<Expr>());
|
|
|
|
RequiresSecondPass = true;
|
|
|
|
}
|
2009-12-16 14:35:08 +08:00
|
|
|
}
|
2009-08-05 05:02:39 +08:00
|
|
|
} else if (InitListExpr *InnerILE
|
2011-10-22 07:02:22 +08:00
|
|
|
= dyn_cast_or_null<InitListExpr>(InitExpr))
|
2009-12-16 14:35:08 +08:00
|
|
|
FillInValueInitializations(ElementEntity, InnerILE, RequiresSecondPass);
|
2009-01-29 05:54:33 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2009-01-29 13:10:57 +08:00
|
|
|
|
2009-12-16 14:35:08 +08:00
|
|
|
InitListChecker::InitListChecker(Sema &S, const InitializedEntity &Entity,
|
2011-09-25 01:48:00 +08:00
|
|
|
InitListExpr *IL, QualType &T,
|
2013-06-06 19:41:05 +08:00
|
|
|
bool VerifyOnly)
|
|
|
|
: SemaRef(S), VerifyOnly(VerifyOnly) {
|
2008-05-02 06:18:59 +08:00
|
|
|
hadError = false;
|
2008-05-20 04:00:43 +08:00
|
|
|
|
2013-09-21 04:10:22 +08:00
|
|
|
FullyStructuredList =
|
|
|
|
getStructuredSubobjectInit(IL, 0, T, 0, 0, IL->getSourceRange());
|
|
|
|
CheckExplicitInitList(Entity, IL, T, FullyStructuredList,
|
2009-02-05 06:46:25 +08:00
|
|
|
/*TopLevelObject=*/true);
|
2008-05-20 04:00:43 +08:00
|
|
|
|
2011-09-25 01:48:00 +08:00
|
|
|
if (!hadError && !VerifyOnly) {
|
2009-12-16 14:35:08 +08:00
|
|
|
bool RequiresSecondPass = false;
|
|
|
|
FillInValueInitializations(Entity, FullyStructuredList, RequiresSecondPass);
|
2009-12-17 02:50:27 +08:00
|
|
|
if (RequiresSecondPass && !hadError)
|
2011-01-27 15:10:08 +08:00
|
|
|
FillInValueInitializations(Entity, FullyStructuredList,
|
2009-12-16 14:35:08 +08:00
|
|
|
RequiresSecondPass);
|
|
|
|
}
|
2008-05-02 06:18:59 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
int InitListChecker::numArrayElements(QualType DeclType) {
|
2008-05-25 21:22:35 +08:00
|
|
|
// FIXME: use a proper constant
|
|
|
|
int maxElements = 0x7FFFFFFF;
|
2008-08-04 15:31:14 +08:00
|
|
|
if (const ConstantArrayType *CAT =
|
2009-02-25 06:50:46 +08:00
|
|
|
SemaRef.Context.getAsConstantArrayType(DeclType)) {
|
2008-05-02 06:18:59 +08:00
|
|
|
maxElements = static_cast<int>(CAT->getSize().getZExtValue());
|
|
|
|
}
|
|
|
|
return maxElements;
|
|
|
|
}
|
|
|
|
|
|
|
|
int InitListChecker::numStructUnionElements(QualType DeclType) {
|
2009-07-30 05:53:49 +08:00
|
|
|
RecordDecl *structDecl = DeclType->getAs<RecordType>()->getDecl();
|
2009-01-29 05:54:33 +08:00
|
|
|
int InitializableMembers = 0;
|
2009-09-09 23:08:12 +08:00
|
|
|
for (RecordDecl::field_iterator
|
2009-06-30 10:36:12 +08:00
|
|
|
Field = structDecl->field_begin(),
|
|
|
|
FieldEnd = structDecl->field_end();
|
2009-01-29 05:54:33 +08:00
|
|
|
Field != FieldEnd; ++Field) {
|
2011-10-11 01:22:13 +08:00
|
|
|
if (!Field->isUnnamedBitfield())
|
2009-01-29 05:54:33 +08:00
|
|
|
++InitializableMembers;
|
|
|
|
}
|
2008-06-10 07:19:58 +08:00
|
|
|
if (structDecl->isUnion())
|
2008-05-25 22:03:31 +08:00
|
|
|
return std::min(InitializableMembers, 1);
|
|
|
|
return InitializableMembers - structDecl->hasFlexibleArrayMember();
|
2008-05-02 06:18:59 +08:00
|
|
|
}
|
|
|
|
|
2013-09-21 04:10:22 +08:00
|
|
|
/// Check whether the range of the initializer \p ParentIList from element
|
|
|
|
/// \p Index onwards can be used to initialize an object of type \p T. Update
|
|
|
|
/// \p Index to indicate how many elements of the list were consumed.
|
|
|
|
///
|
|
|
|
/// This also fills in \p StructuredList, from element \p StructuredIndex
|
|
|
|
/// onwards, with the fully-braced, desugared form of the initialization.
|
2010-01-24 07:23:01 +08:00
|
|
|
void InitListChecker::CheckImplicitInitList(const InitializedEntity &Entity,
|
2010-01-24 04:47:59 +08:00
|
|
|
InitListExpr *ParentIList,
|
2009-01-29 05:54:33 +08:00
|
|
|
QualType T, unsigned &Index,
|
|
|
|
InitListExpr *StructuredList,
|
2011-08-24 04:17:13 +08:00
|
|
|
unsigned &StructuredIndex) {
|
2008-05-02 06:18:59 +08:00
|
|
|
int maxElements = 0;
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2008-05-02 06:18:59 +08:00
|
|
|
if (T->isArrayType())
|
|
|
|
maxElements = numArrayElements(T);
|
2010-04-27 05:31:17 +08:00
|
|
|
else if (T->isRecordType())
|
2008-05-02 06:18:59 +08:00
|
|
|
maxElements = numStructUnionElements(T);
|
2008-05-20 03:16:24 +08:00
|
|
|
else if (T->isVectorType())
|
2009-09-22 07:43:11 +08:00
|
|
|
maxElements = T->getAs<VectorType>()->getNumElements();
|
2008-05-02 06:18:59 +08:00
|
|
|
else
|
2011-09-23 13:06:16 +08:00
|
|
|
llvm_unreachable("CheckImplicitInitList(): Illegal type");
|
2008-05-20 03:16:24 +08:00
|
|
|
|
2008-05-25 21:49:22 +08:00
|
|
|
if (maxElements == 0) {
|
2011-09-25 01:48:00 +08:00
|
|
|
if (!VerifyOnly)
|
|
|
|
SemaRef.Diag(ParentIList->getInit(Index)->getLocStart(),
|
|
|
|
diag::err_implicit_empty_initializer);
|
2009-01-29 05:54:33 +08:00
|
|
|
++Index;
|
2008-05-25 21:49:22 +08:00
|
|
|
hadError = true;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2009-01-29 05:54:33 +08:00
|
|
|
// Build a structured initializer list corresponding to this subobject.
|
|
|
|
InitListExpr *StructuredSubobjectInitList
|
2009-09-09 23:08:12 +08:00
|
|
|
= getStructuredSubobjectInit(ParentIList, Index, T, StructuredList,
|
|
|
|
StructuredIndex,
|
2012-03-10 02:35:03 +08:00
|
|
|
SourceRange(ParentIList->getInit(Index)->getLocStart(),
|
2009-03-02 01:12:46 +08:00
|
|
|
ParentIList->getSourceRange().getEnd()));
|
2009-01-29 05:54:33 +08:00
|
|
|
unsigned StructuredSubobjectInitIndex = 0;
|
|
|
|
|
|
|
|
// Check the element types and build the structural subobject.
|
2009-02-03 01:43:21 +08:00
|
|
|
unsigned StartIndex = Index;
|
2011-01-27 15:10:08 +08:00
|
|
|
CheckListElementTypes(Entity, ParentIList, T,
|
2010-01-24 04:47:59 +08:00
|
|
|
/*SubobjectIsDesignatorContext=*/false, Index,
|
2009-09-09 23:08:12 +08:00
|
|
|
StructuredSubobjectInitList,
|
2011-08-24 04:17:13 +08:00
|
|
|
StructuredSubobjectInitIndex);
|
2011-10-17 02:19:28 +08:00
|
|
|
|
2013-06-06 19:41:05 +08:00
|
|
|
if (!VerifyOnly) {
|
2011-09-25 01:48:00 +08:00
|
|
|
StructuredSubobjectInitList->setType(T);
|
|
|
|
|
2011-10-17 02:19:28 +08:00
|
|
|
unsigned EndIndex = (Index == StartIndex? StartIndex : Index - 1);
|
2011-09-25 01:48:00 +08:00
|
|
|
// Update the structured sub-object initializer so that it's ending
|
|
|
|
// range corresponds with the end of the last initializer it used.
|
|
|
|
if (EndIndex < ParentIList->getNumInits()) {
|
|
|
|
SourceLocation EndLoc
|
|
|
|
= ParentIList->getInit(EndIndex)->getSourceRange().getEnd();
|
|
|
|
StructuredSubobjectInitList->setRBraceLoc(EndLoc);
|
|
|
|
}
|
|
|
|
|
2011-10-17 02:19:28 +08:00
|
|
|
// Complain about missing braces.
|
2011-09-25 01:48:00 +08:00
|
|
|
if (T->isArrayType() || T->isRecordType()) {
|
|
|
|
SemaRef.Diag(StructuredSubobjectInitList->getLocStart(),
|
2013-06-06 19:41:05 +08:00
|
|
|
diag::warn_missing_braces)
|
2011-09-25 01:48:00 +08:00
|
|
|
<< StructuredSubobjectInitList->getSourceRange()
|
|
|
|
<< FixItHint::CreateInsertion(
|
|
|
|
StructuredSubobjectInitList->getLocStart(), "{")
|
|
|
|
<< FixItHint::CreateInsertion(
|
|
|
|
SemaRef.PP.getLocForEndOfToken(
|
2011-01-27 15:10:08 +08:00
|
|
|
StructuredSubobjectInitList->getLocEnd()),
|
2011-09-25 01:48:00 +08:00
|
|
|
"}");
|
|
|
|
}
|
2010-03-07 12:17:15 +08:00
|
|
|
}
|
2008-05-02 06:18:59 +08:00
|
|
|
}
|
|
|
|
|
2013-09-21 04:10:22 +08:00
|
|
|
/// Check whether the initializer \p IList (that was written with explicit
|
|
|
|
/// braces) can be used to initialize an object of type \p T.
|
|
|
|
///
|
|
|
|
/// This also fills in \p StructuredList with the fully-braced, desugared
|
|
|
|
/// form of the initialization.
|
2010-01-24 07:23:01 +08:00
|
|
|
void InitListChecker::CheckExplicitInitList(const InitializedEntity &Entity,
|
2010-01-24 03:55:29 +08:00
|
|
|
InitListExpr *IList, QualType &T,
|
2009-01-29 05:54:33 +08:00
|
|
|
InitListExpr *StructuredList,
|
2009-02-05 06:46:25 +08:00
|
|
|
bool TopLevelObject) {
|
2008-05-20 04:00:43 +08:00
|
|
|
assert(IList->isExplicit() && "Illegal Implicit InitListExpr");
|
2011-09-25 01:48:00 +08:00
|
|
|
if (!VerifyOnly) {
|
|
|
|
SyntacticToSemantic[IList] = StructuredList;
|
|
|
|
StructuredList->setSyntacticForm(IList);
|
|
|
|
}
|
2013-09-21 04:10:22 +08:00
|
|
|
|
|
|
|
unsigned Index = 0, StructuredIndex = 0;
|
2011-01-27 15:10:08 +08:00
|
|
|
CheckListElementTypes(Entity, IList, T, /*SubobjectIsDesignatorContext=*/true,
|
2010-01-24 03:55:29 +08:00
|
|
|
Index, StructuredList, StructuredIndex, TopLevelObject);
|
2011-09-25 01:48:00 +08:00
|
|
|
if (!VerifyOnly) {
|
2012-02-23 10:25:10 +08:00
|
|
|
QualType ExprTy = T;
|
|
|
|
if (!ExprTy->isArrayType())
|
|
|
|
ExprTy = ExprTy.getNonLValueExprType(SemaRef.Context);
|
2011-09-25 01:48:00 +08:00
|
|
|
IList->setType(ExprTy);
|
|
|
|
StructuredList->setType(ExprTy);
|
|
|
|
}
|
2008-05-25 21:22:35 +08:00
|
|
|
if (hadError)
|
|
|
|
return;
|
2008-05-20 04:00:43 +08:00
|
|
|
|
2008-05-25 21:22:35 +08:00
|
|
|
if (Index < IList->getNumInits()) {
|
2008-05-20 04:00:43 +08:00
|
|
|
// We have leftover initializers
|
2011-09-25 01:48:00 +08:00
|
|
|
if (VerifyOnly) {
|
2012-03-11 15:00:24 +08:00
|
|
|
if (SemaRef.getLangOpts().CPlusPlus ||
|
|
|
|
(SemaRef.getLangOpts().OpenCL &&
|
2011-09-25 01:48:00 +08:00
|
|
|
IList->getType()->isVectorType())) {
|
|
|
|
hadError = true;
|
|
|
|
}
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2009-05-30 04:20:05 +08:00
|
|
|
if (StructuredIndex == 1 &&
|
2013-05-16 17:22:40 +08:00
|
|
|
IsStringInit(StructuredList->getInit(0), T, SemaRef.Context) ==
|
|
|
|
SIF_None) {
|
2009-02-19 06:23:55 +08:00
|
|
|
unsigned DK = diag::warn_excess_initializers_in_char_array_initializer;
|
2012-03-11 15:00:24 +08:00
|
|
|
if (SemaRef.getLangOpts().CPlusPlus) {
|
2009-02-19 06:23:55 +08:00
|
|
|
DK = diag::err_excess_initializers_in_char_array_initializer;
|
2009-05-30 04:20:05 +08:00
|
|
|
hadError = true;
|
|
|
|
}
|
2008-05-20 04:12:18 +08:00
|
|
|
// Special-case
|
2009-02-25 06:50:46 +08:00
|
|
|
SemaRef.Diag(IList->getInit(Index)->getLocStart(), DK)
|
2008-11-19 13:27:50 +08:00
|
|
|
<< IList->getInit(Index)->getSourceRange();
|
2008-05-20 13:25:56 +08:00
|
|
|
} else if (!T->isIncompleteType()) {
|
2009-01-31 06:26:29 +08:00
|
|
|
// Don't complain for incomplete types, since we'll get an error
|
|
|
|
// elsewhere
|
2009-02-05 06:46:25 +08:00
|
|
|
QualType CurrentObjectType = StructuredList->getType();
|
2009-09-09 23:08:12 +08:00
|
|
|
int initKind =
|
2009-02-05 06:46:25 +08:00
|
|
|
CurrentObjectType->isArrayType()? 0 :
|
|
|
|
CurrentObjectType->isVectorType()? 1 :
|
|
|
|
CurrentObjectType->isScalarType()? 2 :
|
|
|
|
CurrentObjectType->isUnionType()? 3 :
|
|
|
|
4;
|
2009-02-19 06:23:55 +08:00
|
|
|
|
|
|
|
unsigned DK = diag::warn_excess_initializers;
|
2012-03-11 15:00:24 +08:00
|
|
|
if (SemaRef.getLangOpts().CPlusPlus) {
|
2009-05-30 04:20:05 +08:00
|
|
|
DK = diag::err_excess_initializers;
|
2009-07-08 05:53:06 +08:00
|
|
|
hadError = true;
|
|
|
|
}
|
2012-03-11 15:00:24 +08:00
|
|
|
if (SemaRef.getLangOpts().OpenCL && initKind == 1) {
|
2009-07-08 05:53:06 +08:00
|
|
|
DK = diag::err_excess_initializers;
|
2009-05-30 04:20:05 +08:00
|
|
|
hadError = true;
|
|
|
|
}
|
2009-02-19 06:23:55 +08:00
|
|
|
|
2009-02-25 06:50:46 +08:00
|
|
|
SemaRef.Diag(IList->getInit(Index)->getLocStart(), DK)
|
2009-02-05 06:46:25 +08:00
|
|
|
<< initKind << IList->getInit(Index)->getSourceRange();
|
2008-05-20 04:00:43 +08:00
|
|
|
}
|
|
|
|
}
|
2008-05-20 04:20:43 +08:00
|
|
|
|
2011-09-25 01:48:00 +08:00
|
|
|
if (!VerifyOnly && T->isScalarType() && IList->getNumInits() == 1 &&
|
|
|
|
!TopLevelObject)
|
2009-02-25 06:50:46 +08:00
|
|
|
SemaRef.Diag(IList->getLocStart(), diag::warn_braces_around_scalar_init)
|
2009-04-02 07:51:29 +08:00
|
|
|
<< IList->getSourceRange()
|
2010-04-01 01:46:05 +08:00
|
|
|
<< FixItHint::CreateRemoval(IList->getLocStart())
|
|
|
|
<< FixItHint::CreateRemoval(IList->getLocEnd());
|
2008-05-02 06:18:59 +08:00
|
|
|
}
|
|
|
|
|
2010-01-24 07:23:01 +08:00
|
|
|
void InitListChecker::CheckListElementTypes(const InitializedEntity &Entity,
|
2010-01-24 03:55:29 +08:00
|
|
|
InitListExpr *IList,
|
2009-09-09 23:08:12 +08:00
|
|
|
QualType &DeclType,
|
2009-01-23 07:26:18 +08:00
|
|
|
bool SubobjectIsDesignatorContext,
|
2009-01-29 05:54:33 +08:00
|
|
|
unsigned &Index,
|
|
|
|
InitListExpr *StructuredList,
|
2009-02-05 06:46:25 +08:00
|
|
|
unsigned &StructuredIndex,
|
|
|
|
bool TopLevelObject) {
|
2011-09-20 07:17:44 +08:00
|
|
|
if (DeclType->isAnyComplexType() && SubobjectIsDesignatorContext) {
|
|
|
|
// Explicitly braced initializer for complex type can be real+imaginary
|
|
|
|
// parts.
|
|
|
|
CheckComplexType(Entity, IList, DeclType, Index,
|
|
|
|
StructuredList, StructuredIndex);
|
|
|
|
} else if (DeclType->isScalarType()) {
|
2010-01-24 03:55:29 +08:00
|
|
|
CheckScalarType(Entity, IList, DeclType, Index,
|
|
|
|
StructuredList, StructuredIndex);
|
2008-05-20 04:00:43 +08:00
|
|
|
} else if (DeclType->isVectorType()) {
|
2011-01-27 15:10:08 +08:00
|
|
|
CheckVectorType(Entity, IList, DeclType, Index,
|
2010-01-24 03:55:29 +08:00
|
|
|
StructuredList, StructuredIndex);
|
2012-07-07 16:35:56 +08:00
|
|
|
} else if (DeclType->isRecordType()) {
|
|
|
|
assert(DeclType->isAggregateType() &&
|
|
|
|
"non-aggregate records should be handed in CheckSubElementType");
|
|
|
|
RecordDecl *RD = DeclType->getAs<RecordType>()->getDecl();
|
|
|
|
CheckStructUnionTypes(Entity, IList, DeclType, RD->field_begin(),
|
|
|
|
SubobjectIsDesignatorContext, Index,
|
|
|
|
StructuredList, StructuredIndex,
|
|
|
|
TopLevelObject);
|
|
|
|
} else if (DeclType->isArrayType()) {
|
|
|
|
llvm::APSInt Zero(
|
|
|
|
SemaRef.Context.getTypeSize(SemaRef.Context.getSizeType()),
|
|
|
|
false);
|
|
|
|
CheckArrayType(Entity, IList, DeclType, Zero,
|
|
|
|
SubobjectIsDesignatorContext, Index,
|
|
|
|
StructuredList, StructuredIndex);
|
2008-08-11 00:05:48 +08:00
|
|
|
} else if (DeclType->isVoidType() || DeclType->isFunctionType()) {
|
|
|
|
// This type is invalid, issue a diagnostic.
|
2009-01-29 05:54:33 +08:00
|
|
|
++Index;
|
2011-09-25 01:48:00 +08:00
|
|
|
if (!VerifyOnly)
|
|
|
|
SemaRef.Diag(IList->getLocStart(), diag::err_illegal_initializer_type)
|
|
|
|
<< DeclType;
|
2008-05-20 13:25:56 +08:00
|
|
|
hadError = true;
|
2009-01-31 06:09:00 +08:00
|
|
|
} else if (DeclType->isReferenceType()) {
|
2010-01-24 07:23:01 +08:00
|
|
|
CheckReferenceType(Entity, IList, DeclType, Index,
|
|
|
|
StructuredList, StructuredIndex);
|
2010-05-15 19:32:37 +08:00
|
|
|
} else if (DeclType->isObjCObjectType()) {
|
2011-09-25 01:48:00 +08:00
|
|
|
if (!VerifyOnly)
|
|
|
|
SemaRef.Diag(IList->getLocStart(), diag::err_init_objc_class)
|
|
|
|
<< DeclType;
|
2010-05-04 02:24:37 +08:00
|
|
|
hadError = true;
|
2008-05-02 06:18:59 +08:00
|
|
|
} else {
|
2011-09-25 01:48:00 +08:00
|
|
|
if (!VerifyOnly)
|
|
|
|
SemaRef.Diag(IList->getLocStart(), diag::err_illegal_initializer_type)
|
|
|
|
<< DeclType;
|
2010-05-04 02:24:37 +08:00
|
|
|
hadError = true;
|
2008-05-02 06:18:59 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2010-01-24 07:23:01 +08:00
|
|
|
void InitListChecker::CheckSubElementType(const InitializedEntity &Entity,
|
2010-01-24 03:55:29 +08:00
|
|
|
InitListExpr *IList,
|
2009-09-09 23:08:12 +08:00
|
|
|
QualType ElemType,
|
2009-01-29 05:54:33 +08:00
|
|
|
unsigned &Index,
|
|
|
|
InitListExpr *StructuredList,
|
|
|
|
unsigned &StructuredIndex) {
|
2009-01-29 08:39:20 +08:00
|
|
|
Expr *expr = IList->getInit(Index);
|
2013-05-31 10:56:17 +08:00
|
|
|
|
|
|
|
if (ElemType->isReferenceType())
|
|
|
|
return CheckReferenceType(Entity, IList, ElemType, Index,
|
|
|
|
StructuredList, StructuredIndex);
|
|
|
|
|
2008-05-20 04:00:43 +08:00
|
|
|
if (InitListExpr *SubInitList = dyn_cast<InitListExpr>(expr)) {
|
2012-07-07 16:35:56 +08:00
|
|
|
if (!ElemType->isRecordType() || ElemType->isAggregateType()) {
|
2013-09-21 04:10:22 +08:00
|
|
|
InitListExpr *InnerStructuredList
|
2012-07-07 16:35:56 +08:00
|
|
|
= getStructuredSubobjectInit(IList, Index, ElemType,
|
|
|
|
StructuredList, StructuredIndex,
|
|
|
|
SubInitList->getSourceRange());
|
2013-09-21 04:10:22 +08:00
|
|
|
CheckExplicitInitList(Entity, SubInitList, ElemType,
|
|
|
|
InnerStructuredList);
|
2012-07-07 16:35:56 +08:00
|
|
|
++StructuredIndex;
|
|
|
|
++Index;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
assert(SemaRef.getLangOpts().CPlusPlus &&
|
|
|
|
"non-aggregate records are only possible in C++");
|
|
|
|
// C++ initialization is handled later.
|
|
|
|
}
|
|
|
|
|
2013-08-20 06:12:56 +08:00
|
|
|
// FIXME: Need to handle atomic aggregate types with implicit init lists.
|
|
|
|
if (ElemType->isScalarType() || ElemType->isAtomicType())
|
2011-02-21 15:57:55 +08:00
|
|
|
return CheckScalarType(Entity, IList, ElemType, Index,
|
|
|
|
StructuredList, StructuredIndex);
|
2009-01-31 06:09:00 +08:00
|
|
|
|
2013-08-20 06:12:56 +08:00
|
|
|
assert((ElemType->isRecordType() || ElemType->isVectorType() ||
|
|
|
|
ElemType->isArrayType()) && "Unexpected type");
|
|
|
|
|
2011-02-21 15:57:55 +08:00
|
|
|
if (const ArrayType *arrayType = SemaRef.Context.getAsArrayType(ElemType)) {
|
|
|
|
// arrayType can be incomplete if we're initializing a flexible
|
|
|
|
// array member. There's nothing we can do with the completed
|
|
|
|
// type here, though.
|
2009-01-31 06:09:00 +08:00
|
|
|
|
2013-05-15 19:03:04 +08:00
|
|
|
if (IsStringInit(expr, arrayType, SemaRef.Context) == SIF_None) {
|
2011-09-27 03:09:09 +08:00
|
|
|
if (!VerifyOnly) {
|
2013-05-15 19:03:04 +08:00
|
|
|
CheckStringInit(expr, ElemType, arrayType, SemaRef);
|
|
|
|
UpdateStructuredListElement(StructuredList, StructuredIndex, expr);
|
2011-09-27 03:09:09 +08:00
|
|
|
}
|
2011-02-21 15:57:55 +08:00
|
|
|
++Index;
|
|
|
|
return;
|
2009-01-31 06:09:00 +08:00
|
|
|
}
|
|
|
|
|
2011-02-21 15:57:55 +08:00
|
|
|
// Fall through for subaggregate initialization.
|
|
|
|
|
2012-03-11 15:00:24 +08:00
|
|
|
} else if (SemaRef.getLangOpts().CPlusPlus) {
|
2009-01-31 06:09:00 +08:00
|
|
|
// C++ [dcl.init.aggr]p12:
|
2011-02-21 15:57:55 +08:00
|
|
|
// All implicit type conversions (clause 4) are considered when
|
2011-09-25 01:47:39 +08:00
|
|
|
// initializing the aggregate member with an initializer from
|
2011-02-21 15:57:55 +08:00
|
|
|
// an initializer-list. If the initializer can initialize a
|
|
|
|
// member, the member is initialized. [...]
|
|
|
|
|
|
|
|
// FIXME: Better EqualLoc?
|
|
|
|
InitializationKind Kind =
|
|
|
|
InitializationKind::CreateCopy(expr->getLocStart(), SourceLocation());
|
2013-05-03 23:05:50 +08:00
|
|
|
InitializationSequence Seq(SemaRef, Entity, Kind, expr);
|
2011-02-21 15:57:55 +08:00
|
|
|
|
|
|
|
if (Seq) {
|
2011-09-25 01:48:00 +08:00
|
|
|
if (!VerifyOnly) {
|
2011-12-20 12:00:21 +08:00
|
|
|
ExprResult Result =
|
2013-05-03 23:05:50 +08:00
|
|
|
Seq.Perform(SemaRef, Entity, Kind, expr);
|
2011-12-20 12:00:21 +08:00
|
|
|
if (Result.isInvalid())
|
|
|
|
hadError = true;
|
2011-09-25 01:48:00 +08:00
|
|
|
|
|
|
|
UpdateStructuredListElement(StructuredList, StructuredIndex,
|
2011-12-20 12:00:21 +08:00
|
|
|
Result.takeAs<Expr>());
|
2011-09-25 01:48:00 +08:00
|
|
|
}
|
2011-02-21 15:57:55 +08:00
|
|
|
++Index;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Fall through for subaggregate initialization
|
|
|
|
} else {
|
|
|
|
// C99 6.7.8p13:
|
2009-09-09 23:08:12 +08:00
|
|
|
//
|
2011-02-21 15:57:55 +08:00
|
|
|
// The initializer for a structure or union object that has
|
|
|
|
// automatic storage duration shall be either an initializer
|
|
|
|
// list as described below, or a single expression that has
|
|
|
|
// compatible structure or union type. In the latter case, the
|
|
|
|
// initial value of the object, including unnamed members, is
|
|
|
|
// that of the expression.
|
2011-04-09 02:41:53 +08:00
|
|
|
ExprResult ExprRes = SemaRef.Owned(expr);
|
2011-02-21 15:57:55 +08:00
|
|
|
if ((ElemType->isRecordType() || ElemType->isVectorType()) &&
|
2011-09-25 01:48:00 +08:00
|
|
|
SemaRef.CheckSingleAssignmentConstraints(ElemType, ExprRes,
|
|
|
|
!VerifyOnly)
|
2013-09-17 12:07:04 +08:00
|
|
|
!= Sema::Incompatible) {
|
2011-04-09 02:41:53 +08:00
|
|
|
if (ExprRes.isInvalid())
|
|
|
|
hadError = true;
|
|
|
|
else {
|
|
|
|
ExprRes = SemaRef.DefaultFunctionArrayLvalueConversion(ExprRes.take());
|
2013-05-03 23:05:50 +08:00
|
|
|
if (ExprRes.isInvalid())
|
|
|
|
hadError = true;
|
2011-04-09 02:41:53 +08:00
|
|
|
}
|
|
|
|
UpdateStructuredListElement(StructuredList, StructuredIndex,
|
|
|
|
ExprRes.takeAs<Expr>());
|
2009-01-31 06:09:00 +08:00
|
|
|
++Index;
|
2011-02-21 15:57:55 +08:00
|
|
|
return;
|
2009-01-31 06:09:00 +08:00
|
|
|
}
|
2011-04-09 02:41:53 +08:00
|
|
|
ExprRes.release();
|
2011-02-21 15:57:55 +08:00
|
|
|
// Fall through for subaggregate initialization
|
|
|
|
}
|
|
|
|
|
|
|
|
// C++ [dcl.init.aggr]p12:
|
|
|
|
//
|
|
|
|
// [...] Otherwise, if the member is itself a non-empty
|
|
|
|
// subaggregate, brace elision is assumed and the initializer is
|
|
|
|
// considered for the initialization of the first member of
|
|
|
|
// the subaggregate.
|
2012-03-11 15:00:24 +08:00
|
|
|
if (!SemaRef.getLangOpts().OpenCL &&
|
2011-07-16 07:07:01 +08:00
|
|
|
(ElemType->isAggregateType() || ElemType->isVectorType())) {
|
2011-02-21 15:57:55 +08:00
|
|
|
CheckImplicitInitList(Entity, IList, ElemType, Index, StructuredList,
|
|
|
|
StructuredIndex);
|
|
|
|
++StructuredIndex;
|
|
|
|
} else {
|
2011-09-25 01:48:00 +08:00
|
|
|
if (!VerifyOnly) {
|
|
|
|
// We cannot initialize this element, so let
|
|
|
|
// PerformCopyInitialization produce the appropriate diagnostic.
|
|
|
|
SemaRef.PerformCopyInitialization(Entity, SourceLocation(),
|
|
|
|
SemaRef.Owned(expr),
|
|
|
|
/*TopLevelOfInitList=*/true);
|
|
|
|
}
|
2011-02-21 15:57:55 +08:00
|
|
|
hadError = true;
|
|
|
|
++Index;
|
|
|
|
++StructuredIndex;
|
2009-01-31 06:09:00 +08:00
|
|
|
}
|
2008-05-20 03:16:24 +08:00
|
|
|
}
|
|
|
|
|
2011-09-20 07:17:44 +08:00
|
|
|
void InitListChecker::CheckComplexType(const InitializedEntity &Entity,
|
|
|
|
InitListExpr *IList, QualType DeclType,
|
|
|
|
unsigned &Index,
|
|
|
|
InitListExpr *StructuredList,
|
|
|
|
unsigned &StructuredIndex) {
|
|
|
|
assert(Index == 0 && "Index in explicit init list must be zero");
|
|
|
|
|
|
|
|
// As an extension, clang supports complex initializers, which initialize
|
|
|
|
// a complex number component-wise. When an explicit initializer list for
|
|
|
|
// a complex number contains two two initializers, this extension kicks in:
|
|
|
|
// it exepcts the initializer list to contain two elements convertible to
|
|
|
|
// the element type of the complex type. The first element initializes
|
|
|
|
// the real part, and the second element intitializes the imaginary part.
|
|
|
|
|
|
|
|
if (IList->getNumInits() != 2)
|
|
|
|
return CheckScalarType(Entity, IList, DeclType, Index, StructuredList,
|
|
|
|
StructuredIndex);
|
|
|
|
|
|
|
|
// This is an extension in C. (The builtin _Complex type does not exist
|
|
|
|
// in the C++ standard.)
|
2012-03-11 15:00:24 +08:00
|
|
|
if (!SemaRef.getLangOpts().CPlusPlus && !VerifyOnly)
|
2011-09-20 07:17:44 +08:00
|
|
|
SemaRef.Diag(IList->getLocStart(), diag::ext_complex_component_init)
|
|
|
|
<< IList->getSourceRange();
|
|
|
|
|
|
|
|
// Initialize the complex number.
|
|
|
|
QualType elementType = DeclType->getAs<ComplexType>()->getElementType();
|
|
|
|
InitializedEntity ElementEntity =
|
|
|
|
InitializedEntity::InitializeElement(SemaRef.Context, 0, Entity);
|
|
|
|
|
|
|
|
for (unsigned i = 0; i < 2; ++i) {
|
|
|
|
ElementEntity.setElementIndex(Index);
|
|
|
|
CheckSubElementType(ElementEntity, IList, elementType, Index,
|
|
|
|
StructuredList, StructuredIndex);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2010-01-24 07:23:01 +08:00
|
|
|
void InitListChecker::CheckScalarType(const InitializedEntity &Entity,
|
2010-01-24 03:55:29 +08:00
|
|
|
InitListExpr *IList, QualType DeclType,
|
2009-01-29 08:39:20 +08:00
|
|
|
unsigned &Index,
|
2009-01-29 05:54:33 +08:00
|
|
|
InitListExpr *StructuredList,
|
|
|
|
unsigned &StructuredIndex) {
|
2010-11-11 08:46:36 +08:00
|
|
|
if (Index >= IList->getNumInits()) {
|
2011-10-19 05:39:00 +08:00
|
|
|
if (!VerifyOnly)
|
|
|
|
SemaRef.Diag(IList->getLocStart(),
|
2013-01-02 19:42:31 +08:00
|
|
|
SemaRef.getLangOpts().CPlusPlus11 ?
|
2011-10-19 05:39:00 +08:00
|
|
|
diag::warn_cxx98_compat_empty_scalar_initializer :
|
|
|
|
diag::err_empty_scalar_initializer)
|
|
|
|
<< IList->getSourceRange();
|
2013-01-02 19:42:31 +08:00
|
|
|
hadError = !SemaRef.getLangOpts().CPlusPlus11;
|
2009-01-29 05:54:33 +08:00
|
|
|
++Index;
|
|
|
|
++StructuredIndex;
|
2008-05-20 04:12:18 +08:00
|
|
|
return;
|
2008-05-02 06:18:59 +08:00
|
|
|
}
|
2010-11-11 08:46:36 +08:00
|
|
|
|
|
|
|
Expr *expr = IList->getInit(Index);
|
|
|
|
if (InitListExpr *SubIList = dyn_cast<InitListExpr>(expr)) {
|
2013-11-19 11:41:32 +08:00
|
|
|
// FIXME: This is invalid, and accepting it causes overload resolution
|
|
|
|
// to pick the wrong overload in some corner cases.
|
2011-09-25 01:48:00 +08:00
|
|
|
if (!VerifyOnly)
|
|
|
|
SemaRef.Diag(SubIList->getLocStart(),
|
2013-11-19 11:41:32 +08:00
|
|
|
diag::ext_many_braces_around_scalar_init)
|
2011-09-25 01:48:00 +08:00
|
|
|
<< SubIList->getSourceRange();
|
2010-11-11 08:46:36 +08:00
|
|
|
|
|
|
|
CheckScalarType(Entity, SubIList, DeclType, Index, StructuredList,
|
|
|
|
StructuredIndex);
|
|
|
|
return;
|
|
|
|
} else if (isa<DesignatedInitExpr>(expr)) {
|
2011-09-25 01:48:00 +08:00
|
|
|
if (!VerifyOnly)
|
2012-03-10 02:35:03 +08:00
|
|
|
SemaRef.Diag(expr->getLocStart(),
|
2011-09-25 01:48:00 +08:00
|
|
|
diag::err_designator_for_scalar_init)
|
|
|
|
<< DeclType << expr->getSourceRange();
|
2010-11-11 08:46:36 +08:00
|
|
|
hadError = true;
|
|
|
|
++Index;
|
|
|
|
++StructuredIndex;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2011-09-25 01:48:00 +08:00
|
|
|
if (VerifyOnly) {
|
|
|
|
if (!SemaRef.CanPerformCopyInitialization(Entity, SemaRef.Owned(expr)))
|
|
|
|
hadError = true;
|
|
|
|
++Index;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2010-11-11 08:46:36 +08:00
|
|
|
ExprResult Result =
|
|
|
|
SemaRef.PerformCopyInitialization(Entity, expr->getLocStart(),
|
2011-07-27 07:20:30 +08:00
|
|
|
SemaRef.Owned(expr),
|
|
|
|
/*TopLevelOfInitList=*/true);
|
2010-11-11 08:46:36 +08:00
|
|
|
|
|
|
|
Expr *ResultExpr = 0;
|
|
|
|
|
|
|
|
if (Result.isInvalid())
|
|
|
|
hadError = true; // types weren't compatible.
|
|
|
|
else {
|
|
|
|
ResultExpr = Result.takeAs<Expr>();
|
2011-01-27 15:10:08 +08:00
|
|
|
|
2010-11-11 08:46:36 +08:00
|
|
|
if (ResultExpr != expr) {
|
|
|
|
// The type was promoted, update initializer list.
|
|
|
|
IList->setInit(Index, ResultExpr);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (hadError)
|
|
|
|
++StructuredIndex;
|
|
|
|
else
|
|
|
|
UpdateStructuredListElement(StructuredList, StructuredIndex, ResultExpr);
|
|
|
|
++Index;
|
2008-05-02 06:18:59 +08:00
|
|
|
}
|
|
|
|
|
2010-01-24 07:23:01 +08:00
|
|
|
void InitListChecker::CheckReferenceType(const InitializedEntity &Entity,
|
|
|
|
InitListExpr *IList, QualType DeclType,
|
2009-01-31 06:09:00 +08:00
|
|
|
unsigned &Index,
|
|
|
|
InitListExpr *StructuredList,
|
|
|
|
unsigned &StructuredIndex) {
|
2011-09-25 01:48:00 +08:00
|
|
|
if (Index >= IList->getNumInits()) {
|
2009-05-16 15:39:55 +08:00
|
|
|
// FIXME: It would be wonderful if we could point at the actual member. In
|
|
|
|
// general, it would be useful to pass location information down the stack,
|
|
|
|
// so that we know the location (or decl) of the "current object" being
|
|
|
|
// initialized.
|
2011-09-25 01:48:00 +08:00
|
|
|
if (!VerifyOnly)
|
|
|
|
SemaRef.Diag(IList->getLocStart(),
|
|
|
|
diag::err_init_reference_member_uninitialized)
|
|
|
|
<< DeclType
|
|
|
|
<< IList->getSourceRange();
|
2009-01-31 06:09:00 +08:00
|
|
|
hadError = true;
|
|
|
|
++Index;
|
|
|
|
++StructuredIndex;
|
|
|
|
return;
|
|
|
|
}
|
2011-09-25 01:48:00 +08:00
|
|
|
|
|
|
|
Expr *expr = IList->getInit(Index);
|
2013-01-02 19:42:31 +08:00
|
|
|
if (isa<InitListExpr>(expr) && !SemaRef.getLangOpts().CPlusPlus11) {
|
2011-09-25 01:48:00 +08:00
|
|
|
if (!VerifyOnly)
|
|
|
|
SemaRef.Diag(IList->getLocStart(), diag::err_init_non_aggr_init_list)
|
|
|
|
<< DeclType << IList->getSourceRange();
|
|
|
|
hadError = true;
|
|
|
|
++Index;
|
|
|
|
++StructuredIndex;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (VerifyOnly) {
|
|
|
|
if (!SemaRef.CanPerformCopyInitialization(Entity, SemaRef.Owned(expr)))
|
|
|
|
hadError = true;
|
|
|
|
++Index;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
ExprResult Result =
|
|
|
|
SemaRef.PerformCopyInitialization(Entity, expr->getLocStart(),
|
|
|
|
SemaRef.Owned(expr),
|
|
|
|
/*TopLevelOfInitList=*/true);
|
|
|
|
|
|
|
|
if (Result.isInvalid())
|
|
|
|
hadError = true;
|
|
|
|
|
|
|
|
expr = Result.takeAs<Expr>();
|
|
|
|
IList->setInit(Index, expr);
|
|
|
|
|
|
|
|
if (hadError)
|
|
|
|
++StructuredIndex;
|
|
|
|
else
|
|
|
|
UpdateStructuredListElement(StructuredList, StructuredIndex, expr);
|
|
|
|
++Index;
|
2009-01-31 06:09:00 +08:00
|
|
|
}
|
|
|
|
|
2010-01-24 07:23:01 +08:00
|
|
|
void InitListChecker::CheckVectorType(const InitializedEntity &Entity,
|
2010-01-24 03:55:29 +08:00
|
|
|
InitListExpr *IList, QualType DeclType,
|
2009-01-29 05:54:33 +08:00
|
|
|
unsigned &Index,
|
|
|
|
InitListExpr *StructuredList,
|
|
|
|
unsigned &StructuredIndex) {
|
2010-10-30 08:11:39 +08:00
|
|
|
const VectorType *VT = DeclType->getAs<VectorType>();
|
|
|
|
unsigned maxElements = VT->getNumElements();
|
|
|
|
unsigned numEltsInit = 0;
|
|
|
|
QualType elementType = VT->getElementType();
|
|
|
|
|
2011-10-17 02:19:20 +08:00
|
|
|
if (Index >= IList->getNumInits()) {
|
|
|
|
// Make sure the element type can be value-initialized.
|
|
|
|
if (VerifyOnly)
|
|
|
|
CheckValueInitializable(
|
|
|
|
InitializedEntity::InitializeElement(SemaRef.Context, 0, Entity));
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2012-03-11 15:00:24 +08:00
|
|
|
if (!SemaRef.getLangOpts().OpenCL) {
|
2010-10-30 08:11:39 +08:00
|
|
|
// If the initializing element is a vector, try to copy-initialize
|
|
|
|
// instead of breaking it apart (which is doomed to failure anyway).
|
|
|
|
Expr *Init = IList->getInit(Index);
|
|
|
|
if (!isa<InitListExpr>(Init) && Init->getType()->isVectorType()) {
|
2011-09-25 01:48:00 +08:00
|
|
|
if (VerifyOnly) {
|
|
|
|
if (!SemaRef.CanPerformCopyInitialization(Entity, SemaRef.Owned(Init)))
|
|
|
|
hadError = true;
|
|
|
|
++Index;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2010-10-30 08:11:39 +08:00
|
|
|
ExprResult Result =
|
|
|
|
SemaRef.PerformCopyInitialization(Entity, Init->getLocStart(),
|
2011-07-27 07:20:30 +08:00
|
|
|
SemaRef.Owned(Init),
|
|
|
|
/*TopLevelOfInitList=*/true);
|
2010-10-30 08:11:39 +08:00
|
|
|
|
|
|
|
Expr *ResultExpr = 0;
|
|
|
|
if (Result.isInvalid())
|
|
|
|
hadError = true; // types weren't compatible.
|
|
|
|
else {
|
|
|
|
ResultExpr = Result.takeAs<Expr>();
|
2011-01-27 15:10:08 +08:00
|
|
|
|
2010-10-30 08:11:39 +08:00
|
|
|
if (ResultExpr != Init) {
|
|
|
|
// The type was promoted, update initializer list.
|
|
|
|
IList->setInit(Index, ResultExpr);
|
2009-08-11 07:49:36 +08:00
|
|
|
}
|
|
|
|
}
|
2010-10-30 08:11:39 +08:00
|
|
|
if (hadError)
|
|
|
|
++StructuredIndex;
|
|
|
|
else
|
2011-09-25 01:48:00 +08:00
|
|
|
UpdateStructuredListElement(StructuredList, StructuredIndex,
|
|
|
|
ResultExpr);
|
2010-10-30 08:11:39 +08:00
|
|
|
++Index;
|
|
|
|
return;
|
2008-05-02 06:18:59 +08:00
|
|
|
}
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2010-10-30 08:11:39 +08:00
|
|
|
InitializedEntity ElementEntity =
|
|
|
|
InitializedEntity::InitializeElement(SemaRef.Context, 0, Entity);
|
2011-01-27 15:10:08 +08:00
|
|
|
|
2010-10-30 08:11:39 +08:00
|
|
|
for (unsigned i = 0; i < maxElements; ++i, ++numEltsInit) {
|
|
|
|
// Don't attempt to go past the end of the init list
|
2011-10-17 02:19:20 +08:00
|
|
|
if (Index >= IList->getNumInits()) {
|
|
|
|
if (VerifyOnly)
|
|
|
|
CheckValueInitializable(ElementEntity);
|
2010-10-30 08:11:39 +08:00
|
|
|
break;
|
2011-10-17 02:19:20 +08:00
|
|
|
}
|
2011-01-27 15:10:08 +08:00
|
|
|
|
2010-10-30 08:11:39 +08:00
|
|
|
ElementEntity.setElementIndex(Index);
|
|
|
|
CheckSubElementType(ElementEntity, IList, elementType, Index,
|
|
|
|
StructuredList, StructuredIndex);
|
|
|
|
}
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
InitializedEntity ElementEntity =
|
|
|
|
InitializedEntity::InitializeElement(SemaRef.Context, 0, Entity);
|
2011-01-27 15:10:08 +08:00
|
|
|
|
2010-10-30 08:11:39 +08:00
|
|
|
// OpenCL initializers allows vectors to be constructed from vectors.
|
|
|
|
for (unsigned i = 0; i < maxElements; ++i) {
|
|
|
|
// Don't attempt to go past the end of the init list
|
|
|
|
if (Index >= IList->getNumInits())
|
|
|
|
break;
|
2011-01-27 15:10:08 +08:00
|
|
|
|
2010-10-30 08:11:39 +08:00
|
|
|
ElementEntity.setElementIndex(Index);
|
|
|
|
|
|
|
|
QualType IType = IList->getInit(Index)->getType();
|
|
|
|
if (!IType->isVectorType()) {
|
|
|
|
CheckSubElementType(ElementEntity, IList, elementType, Index,
|
|
|
|
StructuredList, StructuredIndex);
|
|
|
|
++numEltsInit;
|
|
|
|
} else {
|
|
|
|
QualType VecType;
|
|
|
|
const VectorType *IVT = IType->getAs<VectorType>();
|
|
|
|
unsigned numIElts = IVT->getNumElements();
|
2011-01-27 15:10:08 +08:00
|
|
|
|
2010-10-30 08:11:39 +08:00
|
|
|
if (IType->isExtVectorType())
|
|
|
|
VecType = SemaRef.Context.getExtVectorType(elementType, numIElts);
|
|
|
|
else
|
|
|
|
VecType = SemaRef.Context.getVectorType(elementType, numIElts,
|
2010-11-11 05:56:12 +08:00
|
|
|
IVT->getVectorKind());
|
2010-10-30 08:11:39 +08:00
|
|
|
CheckSubElementType(ElementEntity, IList, VecType, Index,
|
|
|
|
StructuredList, StructuredIndex);
|
|
|
|
numEltsInit += numIElts;
|
|
|
|
}
|
2008-05-02 06:18:59 +08:00
|
|
|
}
|
2010-10-30 08:11:39 +08:00
|
|
|
|
|
|
|
// OpenCL requires all elements to be initialized.
|
2011-10-17 02:19:20 +08:00
|
|
|
if (numEltsInit != maxElements) {
|
|
|
|
if (!VerifyOnly)
|
2012-03-10 02:35:03 +08:00
|
|
|
SemaRef.Diag(IList->getLocStart(),
|
2011-10-17 02:19:20 +08:00
|
|
|
diag::err_vector_incorrect_num_initializers)
|
|
|
|
<< (numEltsInit < maxElements) << maxElements << numEltsInit;
|
|
|
|
hadError = true;
|
|
|
|
}
|
2008-05-02 06:18:59 +08:00
|
|
|
}
|
|
|
|
|
2010-01-24 07:23:01 +08:00
|
|
|
void InitListChecker::CheckArrayType(const InitializedEntity &Entity,
|
2010-01-24 04:13:41 +08:00
|
|
|
InitListExpr *IList, QualType &DeclType,
|
2009-01-23 07:26:18 +08:00
|
|
|
llvm::APSInt elementIndex,
|
2009-09-09 23:08:12 +08:00
|
|
|
bool SubobjectIsDesignatorContext,
|
2009-01-29 05:54:33 +08:00
|
|
|
unsigned &Index,
|
|
|
|
InitListExpr *StructuredList,
|
|
|
|
unsigned &StructuredIndex) {
|
2011-02-21 15:22:22 +08:00
|
|
|
const ArrayType *arrayType = SemaRef.Context.getAsArrayType(DeclType);
|
|
|
|
|
2008-05-02 06:18:59 +08:00
|
|
|
// Check for the special-case of initializing an array with a string.
|
|
|
|
if (Index < IList->getNumInits()) {
|
2013-05-15 19:03:04 +08:00
|
|
|
if (IsStringInit(IList->getInit(Index), arrayType, SemaRef.Context) ==
|
|
|
|
SIF_None) {
|
2009-01-29 05:54:33 +08:00
|
|
|
// We place the string literal directly into the resulting
|
|
|
|
// initializer list. This is the only place where the structure
|
|
|
|
// of the structured initializer list doesn't match exactly,
|
|
|
|
// because doing so would involve allocating one character
|
|
|
|
// constant for each string.
|
2011-09-25 01:48:00 +08:00
|
|
|
if (!VerifyOnly) {
|
2013-05-15 19:03:04 +08:00
|
|
|
CheckStringInit(IList->getInit(Index), DeclType, arrayType, SemaRef);
|
|
|
|
UpdateStructuredListElement(StructuredList, StructuredIndex,
|
|
|
|
IList->getInit(Index));
|
2011-09-25 01:48:00 +08:00
|
|
|
StructuredList->resizeInits(SemaRef.Context, StructuredIndex);
|
|
|
|
}
|
2008-05-02 06:18:59 +08:00
|
|
|
++Index;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
2011-02-21 15:22:22 +08:00
|
|
|
if (const VariableArrayType *VAT = dyn_cast<VariableArrayType>(arrayType)) {
|
2008-05-25 21:22:35 +08:00
|
|
|
// Check for VLAs; in standard C it would be possible to check this
|
|
|
|
// earlier, but I don't know where clang accepts VLAs (gcc accepts
|
|
|
|
// them in all sorts of strange places).
|
2011-09-25 01:48:00 +08:00
|
|
|
if (!VerifyOnly)
|
|
|
|
SemaRef.Diag(VAT->getSizeExpr()->getLocStart(),
|
|
|
|
diag::err_variable_object_no_init)
|
|
|
|
<< VAT->getSizeExpr()->getSourceRange();
|
2008-05-25 21:22:35 +08:00
|
|
|
hadError = true;
|
2009-01-29 05:54:33 +08:00
|
|
|
++Index;
|
|
|
|
++StructuredIndex;
|
2008-05-25 21:22:35 +08:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2009-01-22 08:58:24 +08:00
|
|
|
// We might know the maximum number of elements in advance.
|
2009-01-29 05:54:33 +08:00
|
|
|
llvm::APSInt maxElements(elementIndex.getBitWidth(),
|
|
|
|
elementIndex.isUnsigned());
|
2009-01-22 08:58:24 +08:00
|
|
|
bool maxElementsKnown = false;
|
2011-02-21 15:22:22 +08:00
|
|
|
if (const ConstantArrayType *CAT = dyn_cast<ConstantArrayType>(arrayType)) {
|
2009-01-22 08:58:24 +08:00
|
|
|
maxElements = CAT->getSize();
|
2010-12-07 16:25:34 +08:00
|
|
|
elementIndex = elementIndex.extOrTrunc(maxElements.getBitWidth());
|
2009-01-24 02:58:42 +08:00
|
|
|
elementIndex.setIsUnsigned(maxElements.isUnsigned());
|
2009-01-22 08:58:24 +08:00
|
|
|
maxElementsKnown = true;
|
|
|
|
}
|
|
|
|
|
2011-02-21 15:22:22 +08:00
|
|
|
QualType elementType = arrayType->getElementType();
|
2009-01-22 08:58:24 +08:00
|
|
|
while (Index < IList->getNumInits()) {
|
|
|
|
Expr *Init = IList->getInit(Index);
|
|
|
|
if (DesignatedInitExpr *DIE = dyn_cast<DesignatedInitExpr>(Init)) {
|
2009-01-23 07:26:18 +08:00
|
|
|
// If we're not the subobject that matches up with the '{' for
|
|
|
|
// the designator, we shouldn't be handling the
|
|
|
|
// designator. Return immediately.
|
|
|
|
if (!SubobjectIsDesignatorContext)
|
|
|
|
return;
|
|
|
|
|
|
|
|
// Handle this designated initializer. elementIndex will be
|
|
|
|
// updated to be the next array element we'll initialize.
|
2010-01-24 06:49:02 +08:00
|
|
|
if (CheckDesignatedInitializer(Entity, IList, DIE, 0,
|
2009-01-29 05:54:33 +08:00
|
|
|
DeclType, 0, &elementIndex, Index,
|
2009-02-05 06:46:25 +08:00
|
|
|
StructuredList, StructuredIndex, true,
|
|
|
|
false)) {
|
2009-01-22 08:58:24 +08:00
|
|
|
hadError = true;
|
2009-01-23 07:26:18 +08:00
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2009-01-24 00:54:12 +08:00
|
|
|
if (elementIndex.getBitWidth() > maxElements.getBitWidth())
|
2010-12-07 16:25:34 +08:00
|
|
|
maxElements = maxElements.extend(elementIndex.getBitWidth());
|
2009-01-24 00:54:12 +08:00
|
|
|
else if (elementIndex.getBitWidth() < maxElements.getBitWidth())
|
2010-12-07 16:25:34 +08:00
|
|
|
elementIndex = elementIndex.extend(maxElements.getBitWidth());
|
2009-01-24 02:58:42 +08:00
|
|
|
elementIndex.setIsUnsigned(maxElements.isUnsigned());
|
2009-01-24 00:54:12 +08:00
|
|
|
|
2009-01-23 07:26:18 +08:00
|
|
|
// If the array is of incomplete type, keep track of the number of
|
|
|
|
// elements in the initializer.
|
|
|
|
if (!maxElementsKnown && elementIndex > maxElements)
|
|
|
|
maxElements = elementIndex;
|
2009-01-22 08:58:24 +08:00
|
|
|
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
// If we know the maximum number of elements, and we've already
|
|
|
|
// hit it, stop consuming elements in the initializer list.
|
|
|
|
if (maxElementsKnown && elementIndex == maxElements)
|
2008-05-02 06:18:59 +08:00
|
|
|
break;
|
2009-01-22 08:58:24 +08:00
|
|
|
|
2010-01-24 07:23:01 +08:00
|
|
|
InitializedEntity ElementEntity =
|
2011-01-27 15:10:08 +08:00
|
|
|
InitializedEntity::InitializeElement(SemaRef.Context, StructuredIndex,
|
2010-01-24 07:23:01 +08:00
|
|
|
Entity);
|
|
|
|
// Check this element.
|
|
|
|
CheckSubElementType(ElementEntity, IList, elementType, Index,
|
|
|
|
StructuredList, StructuredIndex);
|
2009-01-22 08:58:24 +08:00
|
|
|
++elementIndex;
|
|
|
|
|
|
|
|
// If the array is of incomplete type, keep track of the number of
|
|
|
|
// elements in the initializer.
|
|
|
|
if (!maxElementsKnown && elementIndex > maxElements)
|
|
|
|
maxElements = elementIndex;
|
2008-05-02 06:18:59 +08:00
|
|
|
}
|
2011-09-25 01:48:00 +08:00
|
|
|
if (!hadError && DeclType->isIncompleteArrayType() && !VerifyOnly) {
|
2008-05-02 06:18:59 +08:00
|
|
|
// If this is an incomplete array type, the actual type needs to
|
2008-08-19 04:28:46 +08:00
|
|
|
// be calculated here.
|
2009-01-24 02:58:42 +08:00
|
|
|
llvm::APSInt Zero(maxElements.getBitWidth(), maxElements.isUnsigned());
|
2009-01-22 08:58:24 +08:00
|
|
|
if (maxElements == Zero) {
|
2008-08-19 04:28:46 +08:00
|
|
|
// Sizing an array implicitly to zero is not allowed by ISO C,
|
|
|
|
// but is supported by GNU.
|
2009-02-25 06:50:46 +08:00
|
|
|
SemaRef.Diag(IList->getLocStart(),
|
2008-08-19 04:28:46 +08:00
|
|
|
diag::ext_typecheck_zero_array_size);
|
2008-05-02 06:18:59 +08:00
|
|
|
}
|
2008-08-19 04:28:46 +08:00
|
|
|
|
2009-09-09 23:08:12 +08:00
|
|
|
DeclType = SemaRef.Context.getConstantArrayType(elementType, maxElements,
|
2008-08-19 04:28:46 +08:00
|
|
|
ArrayType::Normal, 0);
|
2008-05-02 06:18:59 +08:00
|
|
|
}
|
2011-10-17 02:19:20 +08:00
|
|
|
if (!hadError && VerifyOnly) {
|
|
|
|
// Check if there are any members of the array that get value-initialized.
|
|
|
|
// If so, check if doing that is possible.
|
|
|
|
// FIXME: This needs to detect holes left by designated initializers too.
|
|
|
|
if (maxElementsKnown && elementIndex < maxElements)
|
|
|
|
CheckValueInitializable(InitializedEntity::InitializeElement(
|
|
|
|
SemaRef.Context, 0, Entity));
|
|
|
|
}
|
2008-05-02 06:18:59 +08:00
|
|
|
}
|
|
|
|
|
2011-08-24 06:24:57 +08:00
|
|
|
bool InitListChecker::CheckFlexibleArrayInit(const InitializedEntity &Entity,
|
|
|
|
Expr *InitExpr,
|
|
|
|
FieldDecl *Field,
|
|
|
|
bool TopLevelObject) {
|
|
|
|
// Handle GNU flexible array initializers.
|
|
|
|
unsigned FlexArrayDiag;
|
|
|
|
if (isa<InitListExpr>(InitExpr) &&
|
|
|
|
cast<InitListExpr>(InitExpr)->getNumInits() == 0) {
|
|
|
|
// Empty flexible array init always allowed as an extension
|
|
|
|
FlexArrayDiag = diag::ext_flexible_array_init;
|
2012-03-11 15:00:24 +08:00
|
|
|
} else if (SemaRef.getLangOpts().CPlusPlus) {
|
2011-08-24 06:24:57 +08:00
|
|
|
// Disallow flexible array init in C++; it is not required for gcc
|
|
|
|
// compatibility, and it needs work to IRGen correctly in general.
|
|
|
|
FlexArrayDiag = diag::err_flexible_array_init;
|
|
|
|
} else if (!TopLevelObject) {
|
|
|
|
// Disallow flexible array init on non-top-level object
|
|
|
|
FlexArrayDiag = diag::err_flexible_array_init;
|
|
|
|
} else if (Entity.getKind() != InitializedEntity::EK_Variable) {
|
|
|
|
// Disallow flexible array init on anything which is not a variable.
|
|
|
|
FlexArrayDiag = diag::err_flexible_array_init;
|
|
|
|
} else if (cast<VarDecl>(Entity.getDecl())->hasLocalStorage()) {
|
|
|
|
// Disallow flexible array init on local variables.
|
|
|
|
FlexArrayDiag = diag::err_flexible_array_init;
|
|
|
|
} else {
|
|
|
|
// Allow other cases.
|
|
|
|
FlexArrayDiag = diag::ext_flexible_array_init;
|
|
|
|
}
|
2011-09-25 01:48:00 +08:00
|
|
|
|
|
|
|
if (!VerifyOnly) {
|
2012-03-10 02:35:03 +08:00
|
|
|
SemaRef.Diag(InitExpr->getLocStart(),
|
2011-09-25 01:48:00 +08:00
|
|
|
FlexArrayDiag)
|
2012-03-10 02:35:03 +08:00
|
|
|
<< InitExpr->getLocStart();
|
2011-09-25 01:48:00 +08:00
|
|
|
SemaRef.Diag(Field->getLocation(), diag::note_flexible_array_member)
|
|
|
|
<< Field;
|
|
|
|
}
|
2011-08-24 06:24:57 +08:00
|
|
|
|
|
|
|
return FlexArrayDiag != diag::ext_flexible_array_init;
|
|
|
|
}
|
|
|
|
|
2010-01-24 07:23:01 +08:00
|
|
|
void InitListChecker::CheckStructUnionTypes(const InitializedEntity &Entity,
|
2010-01-24 04:20:40 +08:00
|
|
|
InitListExpr *IList,
|
2009-09-09 23:08:12 +08:00
|
|
|
QualType DeclType,
|
2009-01-23 07:26:18 +08:00
|
|
|
RecordDecl::field_iterator Field,
|
2009-09-09 23:08:12 +08:00
|
|
|
bool SubobjectIsDesignatorContext,
|
2009-01-29 05:54:33 +08:00
|
|
|
unsigned &Index,
|
|
|
|
InitListExpr *StructuredList,
|
2009-02-05 06:46:25 +08:00
|
|
|
unsigned &StructuredIndex,
|
|
|
|
bool TopLevelObject) {
|
2009-07-30 05:53:49 +08:00
|
|
|
RecordDecl* structDecl = DeclType->getAs<RecordType>()->getDecl();
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2008-05-20 03:16:24 +08:00
|
|
|
// If the record is invalid, some of it's members are invalid. To avoid
|
|
|
|
// confusion, we forgo checking the intializer for the entire record.
|
|
|
|
if (structDecl->isInvalidDecl()) {
|
2012-09-29 05:23:50 +08:00
|
|
|
// Assume it was supposed to consume a single initializer.
|
|
|
|
++Index;
|
2008-05-20 03:16:24 +08:00
|
|
|
hadError = true;
|
|
|
|
return;
|
2009-09-09 23:08:12 +08:00
|
|
|
}
|
2009-01-30 01:44:32 +08:00
|
|
|
|
|
|
|
if (DeclType->isUnionType() && IList->getNumInits() == 0) {
|
2011-10-17 02:19:20 +08:00
|
|
|
RecordDecl *RD = DeclType->getAs<RecordType>()->getDecl();
|
2013-04-21 06:23:05 +08:00
|
|
|
|
|
|
|
// If there's a default initializer, use it.
|
|
|
|
if (isa<CXXRecordDecl>(RD) && cast<CXXRecordDecl>(RD)->hasInClassInitializer()) {
|
|
|
|
if (VerifyOnly)
|
|
|
|
return;
|
|
|
|
for (RecordDecl::field_iterator FieldEnd = RD->field_end();
|
|
|
|
Field != FieldEnd; ++Field) {
|
|
|
|
if (Field->hasInClassInitializer()) {
|
|
|
|
StructuredList->setInitializedFieldInUnion(*Field);
|
|
|
|
// FIXME: Actually build a CXXDefaultInitExpr?
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Value-initialize the first named member of the union.
|
2011-10-17 02:19:20 +08:00
|
|
|
for (RecordDecl::field_iterator FieldEnd = RD->field_end();
|
|
|
|
Field != FieldEnd; ++Field) {
|
|
|
|
if (Field->getDeclName()) {
|
|
|
|
if (VerifyOnly)
|
|
|
|
CheckValueInitializable(
|
2012-06-07 04:45:41 +08:00
|
|
|
InitializedEntity::InitializeMember(*Field, &Entity));
|
2011-10-17 02:19:20 +08:00
|
|
|
else
|
2012-06-07 04:45:41 +08:00
|
|
|
StructuredList->setInitializedFieldInUnion(*Field);
|
2011-10-17 02:19:20 +08:00
|
|
|
break;
|
2009-01-30 01:44:32 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2009-01-22 08:58:24 +08:00
|
|
|
// If structDecl is a forward declaration, this loop won't do
|
|
|
|
// anything except look at designated initializers; That's okay,
|
|
|
|
// because an error should get printed out elsewhere. It might be
|
|
|
|
// worthwhile to skip over the rest of the initializer, though.
|
2009-07-30 05:53:49 +08:00
|
|
|
RecordDecl *RD = DeclType->getAs<RecordType>()->getDecl();
|
2009-06-30 10:36:12 +08:00
|
|
|
RecordDecl::field_iterator FieldEnd = RD->field_end();
|
2009-02-13 03:00:39 +08:00
|
|
|
bool InitializedSomething = false;
|
2010-03-12 03:32:38 +08:00
|
|
|
bool CheckForMissingFields = true;
|
2009-01-22 08:58:24 +08:00
|
|
|
while (Index < IList->getNumInits()) {
|
|
|
|
Expr *Init = IList->getInit(Index);
|
|
|
|
|
|
|
|
if (DesignatedInitExpr *DIE = dyn_cast<DesignatedInitExpr>(Init)) {
|
2009-01-23 07:26:18 +08:00
|
|
|
// If we're not the subobject that matches up with the '{' for
|
|
|
|
// the designator, we shouldn't be handling the
|
|
|
|
// designator. Return immediately.
|
|
|
|
if (!SubobjectIsDesignatorContext)
|
|
|
|
return;
|
|
|
|
|
|
|
|
// Handle this designated initializer. Field will be updated to
|
|
|
|
// the next field that we'll be initializing.
|
2010-01-24 06:49:02 +08:00
|
|
|
if (CheckDesignatedInitializer(Entity, IList, DIE, 0,
|
2009-01-29 05:54:33 +08:00
|
|
|
DeclType, &Field, 0, Index,
|
2009-02-05 06:46:25 +08:00
|
|
|
StructuredList, StructuredIndex,
|
|
|
|
true, TopLevelObject))
|
2009-01-22 08:58:24 +08:00
|
|
|
hadError = true;
|
|
|
|
|
2009-02-13 03:00:39 +08:00
|
|
|
InitializedSomething = true;
|
2010-03-12 03:32:38 +08:00
|
|
|
|
|
|
|
// Disable check for missing fields when designators are used.
|
|
|
|
// This matches gcc behaviour.
|
|
|
|
CheckForMissingFields = false;
|
2009-01-22 08:58:24 +08:00
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (Field == FieldEnd) {
|
|
|
|
// We've run out of fields. We're done.
|
2008-12-12 00:49:14 +08:00
|
|
|
break;
|
2009-01-22 08:58:24 +08:00
|
|
|
}
|
2008-12-12 00:49:14 +08:00
|
|
|
|
2009-02-13 03:00:39 +08:00
|
|
|
// We've already initialized a member of a union. We're done.
|
|
|
|
if (InitializedSomething && DeclType->isUnionType())
|
|
|
|
break;
|
|
|
|
|
2009-01-22 08:58:24 +08:00
|
|
|
// If we've hit the flexible array member at the end, we're done.
|
|
|
|
if (Field->getType()->isIncompleteArrayType())
|
2008-05-20 03:16:24 +08:00
|
|
|
break;
|
2008-12-12 00:49:14 +08:00
|
|
|
|
2009-01-30 00:53:55 +08:00
|
|
|
if (Field->isUnnamedBitfield()) {
|
2009-01-29 05:54:33 +08:00
|
|
|
// Don't initialize unnamed bitfields, e.g. "int : 20;"
|
2009-01-22 08:58:24 +08:00
|
|
|
++Field;
|
2008-05-20 03:16:24 +08:00
|
|
|
continue;
|
2008-05-02 06:18:59 +08:00
|
|
|
}
|
2008-12-12 00:49:14 +08:00
|
|
|
|
2011-06-30 05:51:31 +08:00
|
|
|
// Make sure we can use this declaration.
|
2011-09-25 01:48:00 +08:00
|
|
|
bool InvalidUse;
|
|
|
|
if (VerifyOnly)
|
2012-06-07 04:45:41 +08:00
|
|
|
InvalidUse = !SemaRef.CanUseDecl(*Field);
|
2011-09-25 01:48:00 +08:00
|
|
|
else
|
2012-06-07 04:45:41 +08:00
|
|
|
InvalidUse = SemaRef.DiagnoseUseOfDecl(*Field,
|
2011-09-25 01:48:00 +08:00
|
|
|
IList->getInit(Index)->getLocStart());
|
|
|
|
if (InvalidUse) {
|
2011-06-30 05:51:31 +08:00
|
|
|
++Index;
|
|
|
|
++Field;
|
|
|
|
hadError = true;
|
|
|
|
continue;
|
2011-09-25 01:48:00 +08:00
|
|
|
}
|
2011-06-30 05:51:31 +08:00
|
|
|
|
2010-01-24 07:23:01 +08:00
|
|
|
InitializedEntity MemberEntity =
|
2012-06-07 04:45:41 +08:00
|
|
|
InitializedEntity::InitializeMember(*Field, &Entity);
|
2010-01-24 07:23:01 +08:00
|
|
|
CheckSubElementType(MemberEntity, IList, Field->getType(), Index,
|
|
|
|
StructuredList, StructuredIndex);
|
2009-02-13 03:00:39 +08:00
|
|
|
InitializedSomething = true;
|
2009-01-30 00:53:55 +08:00
|
|
|
|
2011-09-25 01:48:00 +08:00
|
|
|
if (DeclType->isUnionType() && !VerifyOnly) {
|
2009-01-30 00:53:55 +08:00
|
|
|
// Initialize the first field within the union.
|
2012-06-07 04:45:41 +08:00
|
|
|
StructuredList->setInitializedFieldInUnion(*Field);
|
2009-01-30 00:53:55 +08:00
|
|
|
}
|
2009-01-22 08:58:24 +08:00
|
|
|
|
|
|
|
++Field;
|
2008-05-02 06:18:59 +08:00
|
|
|
}
|
2008-12-12 00:49:14 +08:00
|
|
|
|
2010-03-12 03:32:38 +08:00
|
|
|
// Emit warnings for missing struct field initializers.
|
2011-09-25 01:48:00 +08:00
|
|
|
if (!VerifyOnly && InitializedSomething && CheckForMissingFields &&
|
|
|
|
Field != FieldEnd && !Field->getType()->isIncompleteArrayType() &&
|
|
|
|
!DeclType->isUnionType()) {
|
2010-03-12 03:32:38 +08:00
|
|
|
// It is possible we have one or more unnamed bitfields remaining.
|
|
|
|
// Find first (if any) named field and emit warning.
|
|
|
|
for (RecordDecl::field_iterator it = Field, end = RD->field_end();
|
|
|
|
it != end; ++it) {
|
2013-04-21 06:23:05 +08:00
|
|
|
if (!it->isUnnamedBitfield() && !it->hasInClassInitializer()) {
|
2010-03-12 03:32:38 +08:00
|
|
|
SemaRef.Diag(IList->getSourceRange().getEnd(),
|
|
|
|
diag::warn_missing_field_initializers) << it->getName();
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-10-17 02:19:20 +08:00
|
|
|
// Check that any remaining fields can be value-initialized.
|
|
|
|
if (VerifyOnly && Field != FieldEnd && !DeclType->isUnionType() &&
|
|
|
|
!Field->getType()->isIncompleteArrayType()) {
|
|
|
|
// FIXME: Should check for holes left by designated initializers too.
|
|
|
|
for (; Field != FieldEnd && !hadError; ++Field) {
|
2013-04-21 06:23:05 +08:00
|
|
|
if (!Field->isUnnamedBitfield() && !Field->hasInClassInitializer())
|
2011-10-17 02:19:20 +08:00
|
|
|
CheckValueInitializable(
|
2012-06-07 04:45:41 +08:00
|
|
|
InitializedEntity::InitializeMember(*Field, &Entity));
|
2011-10-17 02:19:20 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2009-09-09 23:08:12 +08:00
|
|
|
if (Field == FieldEnd || !Field->getType()->isIncompleteArrayType() ||
|
2009-03-20 08:32:56 +08:00
|
|
|
Index >= IList->getNumInits())
|
2009-02-05 06:46:25 +08:00
|
|
|
return;
|
|
|
|
|
2012-06-07 04:45:41 +08:00
|
|
|
if (CheckFlexibleArrayInit(Entity, IList->getInit(Index), *Field,
|
2011-08-24 06:24:57 +08:00
|
|
|
TopLevelObject)) {
|
2009-02-05 06:46:25 +08:00
|
|
|
hadError = true;
|
2009-03-20 08:32:56 +08:00
|
|
|
++Index;
|
|
|
|
return;
|
2009-02-05 06:46:25 +08:00
|
|
|
}
|
|
|
|
|
2010-01-24 07:23:01 +08:00
|
|
|
InitializedEntity MemberEntity =
|
2012-06-07 04:45:41 +08:00
|
|
|
InitializedEntity::InitializeMember(*Field, &Entity);
|
2011-01-27 15:10:08 +08:00
|
|
|
|
2010-01-24 07:23:01 +08:00
|
|
|
if (isa<InitListExpr>(IList->getInit(Index)))
|
2011-01-27 15:10:08 +08:00
|
|
|
CheckSubElementType(MemberEntity, IList, Field->getType(), Index,
|
2010-01-24 07:23:01 +08:00
|
|
|
StructuredList, StructuredIndex);
|
|
|
|
else
|
2011-01-27 15:10:08 +08:00
|
|
|
CheckImplicitInitList(MemberEntity, IList, Field->getType(), Index,
|
2010-01-24 04:47:59 +08:00
|
|
|
StructuredList, StructuredIndex);
|
2008-05-02 06:18:59 +08:00
|
|
|
}
|
|
|
|
|
2009-04-15 14:41:24 +08:00
|
|
|
/// \brief Expand a field designator that refers to a member of an
|
|
|
|
/// anonymous struct or union into a series of field designators that
|
|
|
|
/// refers to the field within the appropriate subobject.
|
|
|
|
///
|
|
|
|
static void ExpandAnonymousFieldDesignator(Sema &SemaRef,
|
2009-09-09 23:08:12 +08:00
|
|
|
DesignatedInitExpr *DIE,
|
|
|
|
unsigned DesigIdx,
|
2010-12-22 11:46:10 +08:00
|
|
|
IndirectFieldDecl *IndirectField) {
|
2009-04-15 14:41:24 +08:00
|
|
|
typedef DesignatedInitExpr::Designator Designator;
|
|
|
|
|
|
|
|
// Build the replacement designators.
|
2011-07-23 18:55:15 +08:00
|
|
|
SmallVector<Designator, 4> Replacements;
|
2010-12-22 11:46:10 +08:00
|
|
|
for (IndirectFieldDecl::chain_iterator PI = IndirectField->chain_begin(),
|
|
|
|
PE = IndirectField->chain_end(); PI != PE; ++PI) {
|
|
|
|
if (PI + 1 == PE)
|
2009-09-09 23:08:12 +08:00
|
|
|
Replacements.push_back(Designator((IdentifierInfo *)0,
|
2009-04-15 14:41:24 +08:00
|
|
|
DIE->getDesignator(DesigIdx)->getDotLoc(),
|
|
|
|
DIE->getDesignator(DesigIdx)->getFieldLoc()));
|
|
|
|
else
|
|
|
|
Replacements.push_back(Designator((IdentifierInfo *)0, SourceLocation(),
|
|
|
|
SourceLocation()));
|
2010-12-22 11:46:10 +08:00
|
|
|
assert(isa<FieldDecl>(*PI));
|
|
|
|
Replacements.back().setField(cast<FieldDecl>(*PI));
|
2009-04-15 14:41:24 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
// Expand the current designator into the set of replacement
|
|
|
|
// designators, so we have a full subobject path down to where the
|
|
|
|
// member of the anonymous struct/union is actually stored.
|
2010-01-07 07:17:19 +08:00
|
|
|
DIE->ExpandDesignator(SemaRef.Context, DesigIdx, &Replacements[0],
|
2009-04-15 14:41:24 +08:00
|
|
|
&Replacements[0] + Replacements.size());
|
2010-12-22 11:46:10 +08:00
|
|
|
}
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2011-01-27 15:10:08 +08:00
|
|
|
/// \brief Given an implicit anonymous field, search the IndirectField that
|
2010-12-22 11:46:10 +08:00
|
|
|
/// corresponds to FieldName.
|
|
|
|
static IndirectFieldDecl *FindIndirectFieldDesignator(FieldDecl *AnonField,
|
|
|
|
IdentifierInfo *FieldName) {
|
2012-09-11 06:04:26 +08:00
|
|
|
if (!FieldName)
|
|
|
|
return 0;
|
|
|
|
|
2010-12-22 11:46:10 +08:00
|
|
|
assert(AnonField->isAnonymousStructOrUnion());
|
|
|
|
Decl *NextDecl = AnonField->getNextDeclInContext();
|
2012-02-10 06:16:56 +08:00
|
|
|
while (IndirectFieldDecl *IF =
|
|
|
|
dyn_cast_or_null<IndirectFieldDecl>(NextDecl)) {
|
2012-09-11 06:04:26 +08:00
|
|
|
if (FieldName == IF->getAnonField()->getIdentifier())
|
2010-12-22 11:46:10 +08:00
|
|
|
return IF;
|
|
|
|
NextDecl = NextDecl->getNextDeclInContext();
|
2009-04-15 14:41:24 +08:00
|
|
|
}
|
2010-12-22 11:46:10 +08:00
|
|
|
return 0;
|
2009-04-15 14:41:24 +08:00
|
|
|
}
|
|
|
|
|
2011-09-25 01:48:00 +08:00
|
|
|
static DesignatedInitExpr *CloneDesignatedInitExpr(Sema &SemaRef,
|
|
|
|
DesignatedInitExpr *DIE) {
|
|
|
|
unsigned NumIndexExprs = DIE->getNumSubExprs() - 1;
|
|
|
|
SmallVector<Expr*, 4> IndexExprs(NumIndexExprs);
|
|
|
|
for (unsigned I = 0; I < NumIndexExprs; ++I)
|
|
|
|
IndexExprs[I] = DIE->getSubExpr(I + 1);
|
|
|
|
return DesignatedInitExpr::Create(SemaRef.Context, DIE->designators_begin(),
|
2012-08-24 19:54:20 +08:00
|
|
|
DIE->size(), IndexExprs,
|
|
|
|
DIE->getEqualOrColonLoc(),
|
2011-09-25 01:48:00 +08:00
|
|
|
DIE->usesGNUSyntax(), DIE->getInit());
|
|
|
|
}
|
|
|
|
|
2012-01-13 03:27:05 +08:00
|
|
|
namespace {
|
|
|
|
|
|
|
|
// Callback to only accept typo corrections that are for field members of
|
|
|
|
// the given struct or union.
|
|
|
|
class FieldInitializerValidatorCCC : public CorrectionCandidateCallback {
|
|
|
|
public:
|
|
|
|
explicit FieldInitializerValidatorCCC(RecordDecl *RD)
|
|
|
|
: Record(RD) {}
|
|
|
|
|
|
|
|
virtual bool ValidateCandidate(const TypoCorrection &candidate) {
|
|
|
|
FieldDecl *FD = candidate.getCorrectionDeclAs<FieldDecl>();
|
|
|
|
return FD && FD->getDeclContext()->getRedeclContext()->Equals(Record);
|
|
|
|
}
|
|
|
|
|
|
|
|
private:
|
|
|
|
RecordDecl *Record;
|
|
|
|
};
|
|
|
|
|
|
|
|
}
|
|
|
|
|
2009-01-22 08:58:24 +08:00
|
|
|
/// @brief Check the well-formedness of a C99 designated initializer.
|
|
|
|
///
|
|
|
|
/// Determines whether the designated initializer @p DIE, which
|
|
|
|
/// resides at the given @p Index within the initializer list @p
|
|
|
|
/// IList, is well-formed for a current object of type @p DeclType
|
|
|
|
/// (C99 6.7.8). The actual subobject that this designator refers to
|
2009-09-09 23:08:12 +08:00
|
|
|
/// within the current subobject is returned in either
|
2009-01-29 05:54:33 +08:00
|
|
|
/// @p NextField or @p NextElementIndex (whichever is appropriate).
|
2009-01-22 08:58:24 +08:00
|
|
|
///
|
|
|
|
/// @param IList The initializer list in which this designated
|
|
|
|
/// initializer occurs.
|
|
|
|
///
|
2009-04-15 12:56:10 +08:00
|
|
|
/// @param DIE The designated initializer expression.
|
|
|
|
///
|
|
|
|
/// @param DesigIdx The index of the current designator.
|
2009-01-22 08:58:24 +08:00
|
|
|
///
|
2012-08-24 01:58:28 +08:00
|
|
|
/// @param CurrentObjectType The type of the "current object" (C99 6.7.8p17),
|
2009-01-22 08:58:24 +08:00
|
|
|
/// into which the designation in @p DIE should refer.
|
|
|
|
///
|
2009-01-23 07:26:18 +08:00
|
|
|
/// @param NextField If non-NULL and the first designator in @p DIE is
|
|
|
|
/// a field, this will be set to the field declaration corresponding
|
|
|
|
/// to the field named by the designator.
|
2009-01-22 08:58:24 +08:00
|
|
|
///
|
2009-01-23 07:26:18 +08:00
|
|
|
/// @param NextElementIndex If non-NULL and the first designator in @p
|
|
|
|
/// DIE is an array designator or GNU array-range designator, this
|
|
|
|
/// will be set to the last index initialized by this designator.
|
2009-01-22 08:58:24 +08:00
|
|
|
///
|
|
|
|
/// @param Index Index into @p IList where the designated initializer
|
|
|
|
/// @p DIE occurs.
|
|
|
|
///
|
2009-01-29 05:54:33 +08:00
|
|
|
/// @param StructuredList The initializer list expression that
|
|
|
|
/// describes all of the subobject initializers in the order they'll
|
|
|
|
/// actually be initialized.
|
|
|
|
///
|
2009-01-22 08:58:24 +08:00
|
|
|
/// @returns true if there was an error, false otherwise.
|
2009-09-09 23:08:12 +08:00
|
|
|
bool
|
2010-01-24 07:23:01 +08:00
|
|
|
InitListChecker::CheckDesignatedInitializer(const InitializedEntity &Entity,
|
2010-01-24 06:49:02 +08:00
|
|
|
InitListExpr *IList,
|
2011-09-25 01:48:00 +08:00
|
|
|
DesignatedInitExpr *DIE,
|
|
|
|
unsigned DesigIdx,
|
|
|
|
QualType &CurrentObjectType,
|
|
|
|
RecordDecl::field_iterator *NextField,
|
|
|
|
llvm::APSInt *NextElementIndex,
|
|
|
|
unsigned &Index,
|
|
|
|
InitListExpr *StructuredList,
|
|
|
|
unsigned &StructuredIndex,
|
2009-02-05 06:46:25 +08:00
|
|
|
bool FinishSubobjectInit,
|
|
|
|
bool TopLevelObject) {
|
2009-04-15 12:56:10 +08:00
|
|
|
if (DesigIdx == DIE->size()) {
|
2009-01-23 07:26:18 +08:00
|
|
|
// Check the actual initialization for the designated object type.
|
|
|
|
bool prevHadError = hadError;
|
2009-01-29 08:39:20 +08:00
|
|
|
|
|
|
|
// Temporarily remove the designator expression from the
|
|
|
|
// initializer list that the child calls see, so that we don't try
|
|
|
|
// to re-process the designator.
|
|
|
|
unsigned OldIndex = Index;
|
|
|
|
IList->setInit(OldIndex, DIE->getInit());
|
|
|
|
|
2010-01-24 06:49:02 +08:00
|
|
|
CheckSubElementType(Entity, IList, CurrentObjectType, Index,
|
2009-01-29 05:54:33 +08:00
|
|
|
StructuredList, StructuredIndex);
|
2009-01-29 08:39:20 +08:00
|
|
|
|
|
|
|
// Restore the designated initializer expression in the syntactic
|
|
|
|
// form of the initializer list.
|
|
|
|
if (IList->getInit(OldIndex) != DIE->getInit())
|
|
|
|
DIE->setInit(IList->getInit(OldIndex));
|
|
|
|
IList->setInit(OldIndex, DIE);
|
|
|
|
|
2009-01-23 07:26:18 +08:00
|
|
|
return hadError && !prevHadError;
|
|
|
|
}
|
2009-01-22 08:58:24 +08:00
|
|
|
|
2009-04-15 12:56:10 +08:00
|
|
|
DesignatedInitExpr::Designator *D = DIE->getDesignator(DesigIdx);
|
2011-09-25 01:48:00 +08:00
|
|
|
bool IsFirstDesignator = (DesigIdx == 0);
|
|
|
|
if (!VerifyOnly) {
|
|
|
|
assert((IsFirstDesignator || StructuredList) &&
|
|
|
|
"Need a non-designated initializer list to start from");
|
|
|
|
|
|
|
|
// Determine the structural initializer list that corresponds to the
|
|
|
|
// current subobject.
|
2012-02-23 22:48:40 +08:00
|
|
|
StructuredList = IsFirstDesignator? SyntacticToSemantic.lookup(IList)
|
2011-09-25 01:48:00 +08:00
|
|
|
: getStructuredSubobjectInit(IList, Index, CurrentObjectType,
|
|
|
|
StructuredList, StructuredIndex,
|
2012-12-25 22:51:39 +08:00
|
|
|
SourceRange(D->getLocStart(),
|
|
|
|
DIE->getLocEnd()));
|
2011-09-25 01:48:00 +08:00
|
|
|
assert(StructuredList && "Expected a structured initializer list");
|
|
|
|
}
|
2009-01-29 05:54:33 +08:00
|
|
|
|
2009-01-23 07:26:18 +08:00
|
|
|
if (D->isFieldDesignator()) {
|
|
|
|
// C99 6.7.8p7:
|
|
|
|
//
|
|
|
|
// If a designator has the form
|
|
|
|
//
|
|
|
|
// . identifier
|
|
|
|
//
|
|
|
|
// then the current object (defined below) shall have
|
|
|
|
// structure or union type and the identifier shall be the
|
2009-09-09 23:08:12 +08:00
|
|
|
// name of a member of that type.
|
2009-07-30 05:53:49 +08:00
|
|
|
const RecordType *RT = CurrentObjectType->getAs<RecordType>();
|
2009-01-23 07:26:18 +08:00
|
|
|
if (!RT) {
|
|
|
|
SourceLocation Loc = D->getDotLoc();
|
|
|
|
if (Loc.isInvalid())
|
|
|
|
Loc = D->getFieldLoc();
|
2011-09-25 01:48:00 +08:00
|
|
|
if (!VerifyOnly)
|
|
|
|
SemaRef.Diag(Loc, diag::err_field_designator_non_aggr)
|
2012-03-11 15:00:24 +08:00
|
|
|
<< SemaRef.getLangOpts().CPlusPlus << CurrentObjectType;
|
2009-01-23 07:26:18 +08:00
|
|
|
++Index;
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2009-01-29 05:54:33 +08:00
|
|
|
// Note: we perform a linear search of the fields here, despite
|
|
|
|
// the fact that we have a faster lookup method, because we always
|
|
|
|
// need to compute the field's index.
|
2009-04-15 14:41:24 +08:00
|
|
|
FieldDecl *KnownField = D->getField();
|
2009-01-23 07:26:18 +08:00
|
|
|
IdentifierInfo *FieldName = D->getFieldName();
|
2009-01-29 05:54:33 +08:00
|
|
|
unsigned FieldIndex = 0;
|
2009-09-09 23:08:12 +08:00
|
|
|
RecordDecl::field_iterator
|
2009-06-30 10:36:12 +08:00
|
|
|
Field = RT->getDecl()->field_begin(),
|
|
|
|
FieldEnd = RT->getDecl()->field_end();
|
2009-01-29 05:54:33 +08:00
|
|
|
for (; Field != FieldEnd; ++Field) {
|
|
|
|
if (Field->isUnnamedBitfield())
|
|
|
|
continue;
|
2011-01-27 15:10:08 +08:00
|
|
|
|
2010-12-22 11:46:10 +08:00
|
|
|
// If we find a field representing an anonymous field, look in the
|
|
|
|
// IndirectFieldDecl that follow for the designated initializer.
|
|
|
|
if (!KnownField && Field->isAnonymousStructOrUnion()) {
|
|
|
|
if (IndirectFieldDecl *IF =
|
2012-06-07 04:45:41 +08:00
|
|
|
FindIndirectFieldDesignator(*Field, FieldName)) {
|
2011-09-25 01:48:00 +08:00
|
|
|
// In verify mode, don't modify the original.
|
|
|
|
if (VerifyOnly)
|
|
|
|
DIE = CloneDesignatedInitExpr(SemaRef, DIE);
|
2010-12-22 11:46:10 +08:00
|
|
|
ExpandAnonymousFieldDesignator(SemaRef, DIE, DesigIdx, IF);
|
|
|
|
D = DIE->getDesignator(DesigIdx);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
2012-06-07 04:45:41 +08:00
|
|
|
if (KnownField && KnownField == *Field)
|
2010-10-09 04:44:28 +08:00
|
|
|
break;
|
|
|
|
if (FieldName && FieldName == Field->getIdentifier())
|
2009-01-29 05:54:33 +08:00
|
|
|
break;
|
|
|
|
|
|
|
|
++FieldIndex;
|
2009-01-23 07:26:18 +08:00
|
|
|
}
|
|
|
|
|
2009-01-29 05:54:33 +08:00
|
|
|
if (Field == FieldEnd) {
|
2011-09-25 10:41:26 +08:00
|
|
|
if (VerifyOnly) {
|
|
|
|
++Index;
|
2011-09-25 01:48:00 +08:00
|
|
|
return true; // No typo correction when just trying this out.
|
2011-09-25 10:41:26 +08:00
|
|
|
}
|
2011-09-25 01:48:00 +08:00
|
|
|
|
2009-04-15 14:41:24 +08:00
|
|
|
// There was no normal field in the struct with the designated
|
|
|
|
// name. Perform another lookup for this name, which may find
|
|
|
|
// something that we can't designate (e.g., a member function),
|
|
|
|
// may find nothing, or may find a member of an anonymous
|
2009-09-09 23:08:12 +08:00
|
|
|
// struct/union.
|
2009-06-30 10:36:12 +08:00
|
|
|
DeclContext::lookup_result Lookup = RT->getDecl()->lookup(FieldName);
|
2010-01-01 08:03:05 +08:00
|
|
|
FieldDecl *ReplacementField = 0;
|
2012-12-19 08:45:41 +08:00
|
|
|
if (Lookup.empty()) {
|
2010-01-01 08:03:05 +08:00
|
|
|
// Name lookup didn't find anything. Determine whether this
|
|
|
|
// was a typo for another field name.
|
2012-01-13 03:27:05 +08:00
|
|
|
FieldInitializerValidatorCCC Validator(RT->getDecl());
|
2013-08-17 08:46:16 +08:00
|
|
|
if (TypoCorrection Corrected = SemaRef.CorrectTypo(
|
|
|
|
DeclarationNameInfo(FieldName, D->getFieldLoc()),
|
|
|
|
Sema::LookupMemberName, /*Scope=*/ 0, /*SS=*/ 0, Validator,
|
|
|
|
RT->getDecl())) {
|
|
|
|
SemaRef.diagnoseTypo(
|
|
|
|
Corrected,
|
|
|
|
SemaRef.PDiag(diag::err_field_designator_unknown_suggest)
|
|
|
|
<< FieldName << CurrentObjectType);
|
2012-01-13 03:27:05 +08:00
|
|
|
ReplacementField = Corrected.getCorrectionDeclAs<FieldDecl>();
|
2011-09-25 10:41:26 +08:00
|
|
|
hadError = true;
|
2010-01-01 08:03:05 +08:00
|
|
|
} else {
|
|
|
|
SemaRef.Diag(D->getFieldLoc(), diag::err_field_designator_unknown)
|
|
|
|
<< FieldName << CurrentObjectType;
|
|
|
|
++Index;
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
}
|
2011-01-27 15:10:08 +08:00
|
|
|
|
2010-01-01 08:03:05 +08:00
|
|
|
if (!ReplacementField) {
|
2009-01-29 05:54:33 +08:00
|
|
|
// Name lookup found something, but it wasn't a field.
|
2009-02-25 06:50:46 +08:00
|
|
|
SemaRef.Diag(D->getFieldLoc(), diag::err_field_designator_nonfield)
|
2009-01-29 05:54:33 +08:00
|
|
|
<< FieldName;
|
2012-12-19 08:45:41 +08:00
|
|
|
SemaRef.Diag(Lookup.front()->getLocation(),
|
2009-01-29 05:54:33 +08:00
|
|
|
diag::note_field_designator_found);
|
2009-04-17 01:49:48 +08:00
|
|
|
++Index;
|
|
|
|
return true;
|
2009-04-15 14:41:24 +08:00
|
|
|
}
|
2010-01-01 08:03:05 +08:00
|
|
|
|
2010-12-22 11:46:10 +08:00
|
|
|
if (!KnownField) {
|
2010-01-01 08:03:05 +08:00
|
|
|
// The replacement field comes from typo correction; find it
|
|
|
|
// in the list of fields.
|
|
|
|
FieldIndex = 0;
|
|
|
|
Field = RT->getDecl()->field_begin();
|
|
|
|
for (; Field != FieldEnd; ++Field) {
|
|
|
|
if (Field->isUnnamedBitfield())
|
|
|
|
continue;
|
|
|
|
|
2012-06-07 04:45:41 +08:00
|
|
|
if (ReplacementField == *Field ||
|
2010-01-01 08:03:05 +08:00
|
|
|
Field->getIdentifier() == ReplacementField->getIdentifier())
|
|
|
|
break;
|
|
|
|
|
|
|
|
++FieldIndex;
|
|
|
|
}
|
|
|
|
}
|
2009-01-23 07:26:18 +08:00
|
|
|
}
|
2009-01-29 05:54:33 +08:00
|
|
|
|
|
|
|
// All of the fields of a union are located at the same place in
|
|
|
|
// the initializer list.
|
2009-01-30 00:53:55 +08:00
|
|
|
if (RT->getDecl()->isUnion()) {
|
2009-01-29 05:54:33 +08:00
|
|
|
FieldIndex = 0;
|
2013-10-03 20:14:24 +08:00
|
|
|
if (!VerifyOnly) {
|
|
|
|
FieldDecl *CurrentField = StructuredList->getInitializedFieldInUnion();
|
|
|
|
if (CurrentField && CurrentField != *Field) {
|
|
|
|
assert(StructuredList->getNumInits() == 1
|
|
|
|
&& "A union should never have more than one initializer!");
|
|
|
|
|
|
|
|
// we're about to throw away an initializer, emit warning
|
|
|
|
SemaRef.Diag(D->getFieldLoc(),
|
|
|
|
diag::warn_initializer_overrides)
|
|
|
|
<< D->getSourceRange();
|
|
|
|
Expr *ExistingInit = StructuredList->getInit(0);
|
|
|
|
SemaRef.Diag(ExistingInit->getLocStart(),
|
|
|
|
diag::note_previous_initializer)
|
|
|
|
<< /*FIXME:has side effects=*/0
|
|
|
|
<< ExistingInit->getSourceRange();
|
|
|
|
|
|
|
|
// remove existing initializer
|
|
|
|
StructuredList->resizeInits(SemaRef.Context, 0);
|
|
|
|
StructuredList->setInitializedFieldInUnion(0);
|
|
|
|
}
|
|
|
|
|
2012-06-07 04:45:41 +08:00
|
|
|
StructuredList->setInitializedFieldInUnion(*Field);
|
2013-10-03 20:14:24 +08:00
|
|
|
}
|
2009-01-30 00:53:55 +08:00
|
|
|
}
|
2009-01-29 05:54:33 +08:00
|
|
|
|
2011-06-30 05:51:31 +08:00
|
|
|
// Make sure we can use this declaration.
|
2011-09-25 01:48:00 +08:00
|
|
|
bool InvalidUse;
|
|
|
|
if (VerifyOnly)
|
2012-06-07 04:45:41 +08:00
|
|
|
InvalidUse = !SemaRef.CanUseDecl(*Field);
|
2011-09-25 01:48:00 +08:00
|
|
|
else
|
2012-06-07 04:45:41 +08:00
|
|
|
InvalidUse = SemaRef.DiagnoseUseOfDecl(*Field, D->getFieldLoc());
|
2011-09-25 01:48:00 +08:00
|
|
|
if (InvalidUse) {
|
2011-06-30 05:51:31 +08:00
|
|
|
++Index;
|
|
|
|
return true;
|
2011-09-25 01:48:00 +08:00
|
|
|
}
|
2011-06-30 05:51:31 +08:00
|
|
|
|
2011-09-25 01:48:00 +08:00
|
|
|
if (!VerifyOnly) {
|
|
|
|
// Update the designator with the field declaration.
|
2012-06-07 04:45:41 +08:00
|
|
|
D->setField(*Field);
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2011-09-25 01:48:00 +08:00
|
|
|
// Make sure that our non-designated initializer list has space
|
|
|
|
// for a subobject corresponding to this field.
|
|
|
|
if (FieldIndex >= StructuredList->getNumInits())
|
|
|
|
StructuredList->resizeInits(SemaRef.Context, FieldIndex + 1);
|
|
|
|
}
|
2009-01-29 05:54:33 +08:00
|
|
|
|
2009-02-05 06:46:25 +08:00
|
|
|
// This designator names a flexible array member.
|
|
|
|
if (Field->getType()->isIncompleteArrayType()) {
|
|
|
|
bool Invalid = false;
|
2009-04-15 12:56:10 +08:00
|
|
|
if ((DesigIdx + 1) != DIE->size()) {
|
2009-02-05 06:46:25 +08:00
|
|
|
// We can't designate an object within the flexible array
|
|
|
|
// member (because GCC doesn't allow it).
|
2011-09-25 01:48:00 +08:00
|
|
|
if (!VerifyOnly) {
|
|
|
|
DesignatedInitExpr::Designator *NextD
|
|
|
|
= DIE->getDesignator(DesigIdx + 1);
|
2012-12-25 22:51:39 +08:00
|
|
|
SemaRef.Diag(NextD->getLocStart(),
|
2011-09-25 01:48:00 +08:00
|
|
|
diag::err_designator_into_flexible_array_member)
|
2012-12-25 22:51:39 +08:00
|
|
|
<< SourceRange(NextD->getLocStart(),
|
|
|
|
DIE->getLocEnd());
|
2011-09-25 01:48:00 +08:00
|
|
|
SemaRef.Diag(Field->getLocation(), diag::note_flexible_array_member)
|
2012-06-07 04:45:41 +08:00
|
|
|
<< *Field;
|
2011-09-25 01:48:00 +08:00
|
|
|
}
|
2009-02-05 06:46:25 +08:00
|
|
|
Invalid = true;
|
|
|
|
}
|
|
|
|
|
2010-10-11 01:49:49 +08:00
|
|
|
if (!hadError && !isa<InitListExpr>(DIE->getInit()) &&
|
|
|
|
!isa<StringLiteral>(DIE->getInit())) {
|
2009-02-05 06:46:25 +08:00
|
|
|
// The initializer is not an initializer list.
|
2011-09-25 01:48:00 +08:00
|
|
|
if (!VerifyOnly) {
|
2012-03-10 02:35:03 +08:00
|
|
|
SemaRef.Diag(DIE->getInit()->getLocStart(),
|
2011-09-25 01:48:00 +08:00
|
|
|
diag::err_flexible_array_init_needs_braces)
|
|
|
|
<< DIE->getInit()->getSourceRange();
|
|
|
|
SemaRef.Diag(Field->getLocation(), diag::note_flexible_array_member)
|
2012-06-07 04:45:41 +08:00
|
|
|
<< *Field;
|
2011-09-25 01:48:00 +08:00
|
|
|
}
|
2009-02-05 06:46:25 +08:00
|
|
|
Invalid = true;
|
|
|
|
}
|
|
|
|
|
2011-08-24 06:24:57 +08:00
|
|
|
// Check GNU flexible array initializer.
|
2012-06-07 04:45:41 +08:00
|
|
|
if (!Invalid && CheckFlexibleArrayInit(Entity, DIE->getInit(), *Field,
|
2011-08-24 06:24:57 +08:00
|
|
|
TopLevelObject))
|
2009-02-05 06:46:25 +08:00
|
|
|
Invalid = true;
|
|
|
|
|
|
|
|
if (Invalid) {
|
|
|
|
++Index;
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Initialize the array.
|
|
|
|
bool prevHadError = hadError;
|
|
|
|
unsigned newStructuredIndex = FieldIndex;
|
|
|
|
unsigned OldIndex = Index;
|
|
|
|
IList->setInit(Index, DIE->getInit());
|
2010-01-24 07:23:01 +08:00
|
|
|
|
|
|
|
InitializedEntity MemberEntity =
|
2012-06-07 04:45:41 +08:00
|
|
|
InitializedEntity::InitializeMember(*Field, &Entity);
|
2010-01-24 07:23:01 +08:00
|
|
|
CheckSubElementType(MemberEntity, IList, Field->getType(), Index,
|
2009-02-05 06:46:25 +08:00
|
|
|
StructuredList, newStructuredIndex);
|
2010-01-24 07:23:01 +08:00
|
|
|
|
2009-02-05 06:46:25 +08:00
|
|
|
IList->setInit(OldIndex, DIE);
|
|
|
|
if (hadError && !prevHadError) {
|
|
|
|
++Field;
|
|
|
|
++FieldIndex;
|
|
|
|
if (NextField)
|
|
|
|
*NextField = Field;
|
|
|
|
StructuredIndex = FieldIndex;
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
// Recurse to check later designated subobjects.
|
2012-04-30 10:36:29 +08:00
|
|
|
QualType FieldType = Field->getType();
|
2009-02-05 06:46:25 +08:00
|
|
|
unsigned newStructuredIndex = FieldIndex;
|
2011-01-27 15:10:08 +08:00
|
|
|
|
2010-01-24 06:49:02 +08:00
|
|
|
InitializedEntity MemberEntity =
|
2012-06-07 04:45:41 +08:00
|
|
|
InitializedEntity::InitializeMember(*Field, &Entity);
|
2011-01-27 15:10:08 +08:00
|
|
|
if (CheckDesignatedInitializer(MemberEntity, IList, DIE, DesigIdx + 1,
|
|
|
|
FieldType, 0, 0, Index,
|
2010-01-24 06:49:02 +08:00
|
|
|
StructuredList, newStructuredIndex,
|
2009-02-05 06:46:25 +08:00
|
|
|
true, false))
|
|
|
|
return true;
|
|
|
|
}
|
2009-01-23 07:26:18 +08:00
|
|
|
|
|
|
|
// Find the position of the next field to be initialized in this
|
|
|
|
// subobject.
|
|
|
|
++Field;
|
2009-01-29 05:54:33 +08:00
|
|
|
++FieldIndex;
|
2009-01-22 08:58:24 +08:00
|
|
|
|
2009-01-23 07:26:18 +08:00
|
|
|
// If this the first designator, our caller will continue checking
|
|
|
|
// the rest of this struct/class/union subobject.
|
|
|
|
if (IsFirstDesignator) {
|
|
|
|
if (NextField)
|
|
|
|
*NextField = Field;
|
2009-01-29 05:54:33 +08:00
|
|
|
StructuredIndex = FieldIndex;
|
2009-01-23 07:26:18 +08:00
|
|
|
return false;
|
|
|
|
}
|
2009-01-22 08:58:24 +08:00
|
|
|
|
2009-01-29 07:36:17 +08:00
|
|
|
if (!FinishSubobjectInit)
|
|
|
|
return false;
|
|
|
|
|
2009-04-15 14:41:24 +08:00
|
|
|
// We've already initialized something in the union; we're done.
|
|
|
|
if (RT->getDecl()->isUnion())
|
|
|
|
return hadError;
|
|
|
|
|
2009-01-23 07:26:18 +08:00
|
|
|
// Check the remaining fields within this class/struct/union subobject.
|
|
|
|
bool prevHadError = hadError;
|
2011-01-27 15:10:08 +08:00
|
|
|
|
2010-01-24 07:23:01 +08:00
|
|
|
CheckStructUnionTypes(Entity, IList, CurrentObjectType, Field, false, Index,
|
2009-01-29 05:54:33 +08:00
|
|
|
StructuredList, FieldIndex);
|
2009-01-23 07:26:18 +08:00
|
|
|
return hadError && !prevHadError;
|
|
|
|
}
|
2009-01-22 08:58:24 +08:00
|
|
|
|
2009-01-23 07:26:18 +08:00
|
|
|
// C99 6.7.8p6:
|
|
|
|
//
|
|
|
|
// If a designator has the form
|
|
|
|
//
|
|
|
|
// [ constant-expression ]
|
|
|
|
//
|
|
|
|
// then the current object (defined below) shall have array
|
|
|
|
// type and the expression shall be an integer constant
|
|
|
|
// expression. If the array is of unknown size, any
|
|
|
|
// nonnegative value is valid.
|
|
|
|
//
|
|
|
|
// Additionally, cope with the GNU extension that permits
|
|
|
|
// designators of the form
|
|
|
|
//
|
|
|
|
// [ constant-expression ... constant-expression ]
|
2009-02-25 06:50:46 +08:00
|
|
|
const ArrayType *AT = SemaRef.Context.getAsArrayType(CurrentObjectType);
|
2009-01-23 07:26:18 +08:00
|
|
|
if (!AT) {
|
2011-09-25 01:48:00 +08:00
|
|
|
if (!VerifyOnly)
|
|
|
|
SemaRef.Diag(D->getLBracketLoc(), diag::err_array_designator_non_array)
|
|
|
|
<< CurrentObjectType;
|
2009-01-23 07:26:18 +08:00
|
|
|
++Index;
|
|
|
|
return true;
|
|
|
|
}
|
2009-01-22 08:58:24 +08:00
|
|
|
|
2009-01-23 07:26:18 +08:00
|
|
|
Expr *IndexExpr = 0;
|
2009-01-29 07:36:17 +08:00
|
|
|
llvm::APSInt DesignatedStartIndex, DesignatedEndIndex;
|
|
|
|
if (D->isArrayDesignator()) {
|
2009-01-23 07:26:18 +08:00
|
|
|
IndexExpr = DIE->getArrayIndex(*D);
|
2011-10-11 02:28:20 +08:00
|
|
|
DesignatedStartIndex = IndexExpr->EvaluateKnownConstInt(SemaRef.Context);
|
2009-01-29 07:36:17 +08:00
|
|
|
DesignatedEndIndex = DesignatedStartIndex;
|
|
|
|
} else {
|
2009-01-23 07:26:18 +08:00
|
|
|
assert(D->isArrayRangeDesignator() && "Need array-range designator");
|
2009-04-26 05:59:05 +08:00
|
|
|
|
2009-09-09 23:08:12 +08:00
|
|
|
DesignatedStartIndex =
|
2011-10-11 02:28:20 +08:00
|
|
|
DIE->getArrayRangeStart(*D)->EvaluateKnownConstInt(SemaRef.Context);
|
2009-09-09 23:08:12 +08:00
|
|
|
DesignatedEndIndex =
|
2011-10-11 02:28:20 +08:00
|
|
|
DIE->getArrayRangeEnd(*D)->EvaluateKnownConstInt(SemaRef.Context);
|
2009-01-23 07:26:18 +08:00
|
|
|
IndexExpr = DIE->getArrayRangeEnd(*D);
|
2009-01-29 07:36:17 +08:00
|
|
|
|
2011-02-20 06:28:58 +08:00
|
|
|
// Codegen can't handle evaluating array range designators that have side
|
|
|
|
// effects, because we replicate the AST value for each initialized element.
|
|
|
|
// As such, set the sawArrayRangeDesignator() bit if we initialize multiple
|
|
|
|
// elements with something that has a side effect, so codegen can emit an
|
|
|
|
// "error unsupported" error instead of miscompiling the app.
|
|
|
|
if (DesignatedStartIndex.getZExtValue()!=DesignatedEndIndex.getZExtValue()&&
|
2011-09-25 01:48:00 +08:00
|
|
|
DIE->getInit()->HasSideEffects(SemaRef.Context) && !VerifyOnly)
|
2009-01-30 03:42:23 +08:00
|
|
|
FullyStructuredList->sawArrayRangeDesignator();
|
2009-01-23 07:26:18 +08:00
|
|
|
}
|
2009-01-22 08:58:24 +08:00
|
|
|
|
2009-01-23 07:26:18 +08:00
|
|
|
if (isa<ConstantArrayType>(AT)) {
|
|
|
|
llvm::APSInt MaxElements(cast<ConstantArrayType>(AT)->getSize(), false);
|
2010-12-07 16:25:34 +08:00
|
|
|
DesignatedStartIndex
|
|
|
|
= DesignatedStartIndex.extOrTrunc(MaxElements.getBitWidth());
|
2009-01-29 07:36:17 +08:00
|
|
|
DesignatedStartIndex.setIsUnsigned(MaxElements.isUnsigned());
|
2010-12-07 16:25:34 +08:00
|
|
|
DesignatedEndIndex
|
|
|
|
= DesignatedEndIndex.extOrTrunc(MaxElements.getBitWidth());
|
2009-01-29 07:36:17 +08:00
|
|
|
DesignatedEndIndex.setIsUnsigned(MaxElements.isUnsigned());
|
|
|
|
if (DesignatedEndIndex >= MaxElements) {
|
2011-09-27 02:53:43 +08:00
|
|
|
if (!VerifyOnly)
|
2012-03-10 02:35:03 +08:00
|
|
|
SemaRef.Diag(IndexExpr->getLocStart(),
|
2011-09-25 01:48:00 +08:00
|
|
|
diag::err_array_designator_too_large)
|
|
|
|
<< DesignatedEndIndex.toString(10) << MaxElements.toString(10)
|
|
|
|
<< IndexExpr->getSourceRange();
|
2009-01-23 07:26:18 +08:00
|
|
|
++Index;
|
|
|
|
return true;
|
2009-01-22 08:58:24 +08:00
|
|
|
}
|
2009-01-29 07:36:17 +08:00
|
|
|
} else {
|
|
|
|
// Make sure the bit-widths and signedness match.
|
|
|
|
if (DesignatedStartIndex.getBitWidth() > DesignatedEndIndex.getBitWidth())
|
2010-12-07 16:25:34 +08:00
|
|
|
DesignatedEndIndex
|
|
|
|
= DesignatedEndIndex.extend(DesignatedStartIndex.getBitWidth());
|
2009-04-26 05:59:05 +08:00
|
|
|
else if (DesignatedStartIndex.getBitWidth() <
|
|
|
|
DesignatedEndIndex.getBitWidth())
|
2010-12-07 16:25:34 +08:00
|
|
|
DesignatedStartIndex
|
|
|
|
= DesignatedStartIndex.extend(DesignatedEndIndex.getBitWidth());
|
2009-01-29 07:36:17 +08:00
|
|
|
DesignatedStartIndex.setIsUnsigned(true);
|
|
|
|
DesignatedEndIndex.setIsUnsigned(true);
|
2009-01-22 08:58:24 +08:00
|
|
|
}
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2013-06-12 05:48:11 +08:00
|
|
|
if (!VerifyOnly && StructuredList->isStringLiteralInit()) {
|
|
|
|
// We're modifying a string literal init; we have to decompose the string
|
|
|
|
// so we can modify the individual characters.
|
|
|
|
ASTContext &Context = SemaRef.Context;
|
|
|
|
Expr *SubExpr = StructuredList->getInit(0)->IgnoreParens();
|
|
|
|
|
|
|
|
// Compute the character type
|
|
|
|
QualType CharTy = AT->getElementType();
|
|
|
|
|
|
|
|
// Compute the type of the integer literals.
|
|
|
|
QualType PromotedCharTy = CharTy;
|
|
|
|
if (CharTy->isPromotableIntegerType())
|
|
|
|
PromotedCharTy = Context.getPromotedIntegerType(CharTy);
|
|
|
|
unsigned PromotedCharTyWidth = Context.getTypeSize(PromotedCharTy);
|
|
|
|
|
|
|
|
if (StringLiteral *SL = dyn_cast<StringLiteral>(SubExpr)) {
|
|
|
|
// Get the length of the string.
|
|
|
|
uint64_t StrLen = SL->getLength();
|
|
|
|
if (cast<ConstantArrayType>(AT)->getSize().ult(StrLen))
|
|
|
|
StrLen = cast<ConstantArrayType>(AT)->getSize().getZExtValue();
|
|
|
|
StructuredList->resizeInits(Context, StrLen);
|
|
|
|
|
|
|
|
// Build a literal for each character in the string, and put them into
|
|
|
|
// the init list.
|
|
|
|
for (unsigned i = 0, e = StrLen; i != e; ++i) {
|
|
|
|
llvm::APInt CodeUnit(PromotedCharTyWidth, SL->getCodeUnit(i));
|
|
|
|
Expr *Init = new (Context) IntegerLiteral(
|
2013-06-12 06:26:34 +08:00
|
|
|
Context, CodeUnit, PromotedCharTy, SubExpr->getExprLoc());
|
2013-06-12 05:48:11 +08:00
|
|
|
if (CharTy != PromotedCharTy)
|
|
|
|
Init = ImplicitCastExpr::Create(Context, CharTy, CK_IntegralCast,
|
|
|
|
Init, 0, VK_RValue);
|
|
|
|
StructuredList->updateInit(Context, i, Init);
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
ObjCEncodeExpr *E = cast<ObjCEncodeExpr>(SubExpr);
|
|
|
|
std::string Str;
|
|
|
|
Context.getObjCEncodingForType(E->getEncodedType(), Str);
|
|
|
|
|
|
|
|
// Get the length of the string.
|
|
|
|
uint64_t StrLen = Str.size();
|
|
|
|
if (cast<ConstantArrayType>(AT)->getSize().ult(StrLen))
|
|
|
|
StrLen = cast<ConstantArrayType>(AT)->getSize().getZExtValue();
|
|
|
|
StructuredList->resizeInits(Context, StrLen);
|
|
|
|
|
|
|
|
// Build a literal for each character in the string, and put them into
|
|
|
|
// the init list.
|
|
|
|
for (unsigned i = 0, e = StrLen; i != e; ++i) {
|
|
|
|
llvm::APInt CodeUnit(PromotedCharTyWidth, Str[i]);
|
|
|
|
Expr *Init = new (Context) IntegerLiteral(
|
2013-06-12 06:26:34 +08:00
|
|
|
Context, CodeUnit, PromotedCharTy, SubExpr->getExprLoc());
|
2013-06-12 05:48:11 +08:00
|
|
|
if (CharTy != PromotedCharTy)
|
|
|
|
Init = ImplicitCastExpr::Create(Context, CharTy, CK_IntegralCast,
|
|
|
|
Init, 0, VK_RValue);
|
|
|
|
StructuredList->updateInit(Context, i, Init);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2009-01-29 05:54:33 +08:00
|
|
|
// Make sure that our non-designated initializer list has space
|
|
|
|
// for a subobject corresponding to this array element.
|
2011-09-25 01:48:00 +08:00
|
|
|
if (!VerifyOnly &&
|
|
|
|
DesignatedEndIndex.getZExtValue() >= StructuredList->getNumInits())
|
2009-09-09 23:08:12 +08:00
|
|
|
StructuredList->resizeInits(SemaRef.Context,
|
2009-01-29 07:36:17 +08:00
|
|
|
DesignatedEndIndex.getZExtValue() + 1);
|
|
|
|
|
|
|
|
// Repeatedly perform subobject initializations in the range
|
|
|
|
// [DesignatedStartIndex, DesignatedEndIndex].
|
|
|
|
|
|
|
|
// Move to the next designator
|
|
|
|
unsigned ElementIndex = DesignatedStartIndex.getZExtValue();
|
|
|
|
unsigned OldIndex = Index;
|
2011-01-27 15:10:08 +08:00
|
|
|
|
2010-01-24 06:49:02 +08:00
|
|
|
InitializedEntity ElementEntity =
|
2010-01-24 07:23:01 +08:00
|
|
|
InitializedEntity::InitializeElement(SemaRef.Context, 0, Entity);
|
2010-01-24 06:49:02 +08:00
|
|
|
|
2009-01-29 07:36:17 +08:00
|
|
|
while (DesignatedStartIndex <= DesignatedEndIndex) {
|
|
|
|
// Recurse to check later designated subobjects.
|
|
|
|
QualType ElementType = AT->getElementType();
|
|
|
|
Index = OldIndex;
|
2011-01-27 15:10:08 +08:00
|
|
|
|
2010-01-24 06:49:02 +08:00
|
|
|
ElementEntity.setElementIndex(ElementIndex);
|
2011-01-27 15:10:08 +08:00
|
|
|
if (CheckDesignatedInitializer(ElementEntity, IList, DIE, DesigIdx + 1,
|
|
|
|
ElementType, 0, 0, Index,
|
2010-01-24 06:49:02 +08:00
|
|
|
StructuredList, ElementIndex,
|
2009-02-05 06:46:25 +08:00
|
|
|
(DesignatedStartIndex == DesignatedEndIndex),
|
|
|
|
false))
|
2009-01-29 07:36:17 +08:00
|
|
|
return true;
|
2009-01-23 07:26:18 +08:00
|
|
|
|
2009-01-29 07:36:17 +08:00
|
|
|
// Move to the next index in the array that we'll be initializing.
|
|
|
|
++DesignatedStartIndex;
|
|
|
|
ElementIndex = DesignatedStartIndex.getZExtValue();
|
|
|
|
}
|
2009-01-23 07:26:18 +08:00
|
|
|
|
|
|
|
// If this the first designator, our caller will continue checking
|
|
|
|
// the rest of this array subobject.
|
|
|
|
if (IsFirstDesignator) {
|
|
|
|
if (NextElementIndex)
|
2009-01-29 07:36:17 +08:00
|
|
|
*NextElementIndex = DesignatedStartIndex;
|
2009-01-29 05:54:33 +08:00
|
|
|
StructuredIndex = ElementIndex;
|
2009-01-23 07:26:18 +08:00
|
|
|
return false;
|
|
|
|
}
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-01-29 07:36:17 +08:00
|
|
|
if (!FinishSubobjectInit)
|
|
|
|
return false;
|
|
|
|
|
2009-01-23 07:26:18 +08:00
|
|
|
// Check the remaining elements within this array subobject.
|
2009-01-22 08:58:24 +08:00
|
|
|
bool prevHadError = hadError;
|
2011-01-27 15:10:08 +08:00
|
|
|
CheckArrayType(Entity, IList, CurrentObjectType, DesignatedStartIndex,
|
2010-01-24 04:13:41 +08:00
|
|
|
/*SubobjectIsDesignatorContext=*/false, Index,
|
2009-01-29 05:54:33 +08:00
|
|
|
StructuredList, ElementIndex);
|
2009-09-09 23:08:12 +08:00
|
|
|
return hadError && !prevHadError;
|
2009-01-22 08:58:24 +08:00
|
|
|
}
|
|
|
|
|
2009-01-29 05:54:33 +08:00
|
|
|
// Get the structured initializer list for a subobject of type
|
|
|
|
// @p CurrentObjectType.
|
|
|
|
InitListExpr *
|
|
|
|
InitListChecker::getStructuredSubobjectInit(InitListExpr *IList, unsigned Index,
|
|
|
|
QualType CurrentObjectType,
|
|
|
|
InitListExpr *StructuredList,
|
|
|
|
unsigned StructuredIndex,
|
|
|
|
SourceRange InitRange) {
|
2011-09-25 01:48:00 +08:00
|
|
|
if (VerifyOnly)
|
|
|
|
return 0; // No structured list in verification-only mode.
|
2009-01-29 05:54:33 +08:00
|
|
|
Expr *ExistingInit = 0;
|
|
|
|
if (!StructuredList)
|
2012-02-23 22:48:40 +08:00
|
|
|
ExistingInit = SyntacticToSemantic.lookup(IList);
|
2009-01-29 05:54:33 +08:00
|
|
|
else if (StructuredIndex < StructuredList->getNumInits())
|
|
|
|
ExistingInit = StructuredList->getInit(StructuredIndex);
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-01-29 05:54:33 +08:00
|
|
|
if (InitListExpr *Result = dyn_cast_or_null<InitListExpr>(ExistingInit))
|
|
|
|
return Result;
|
|
|
|
|
|
|
|
if (ExistingInit) {
|
|
|
|
// We are creating an initializer list that initializes the
|
|
|
|
// subobjects of the current object, but there was already an
|
|
|
|
// initialization that completely initialized the current
|
|
|
|
// subobject, e.g., by a compound literal:
|
2009-09-09 23:08:12 +08:00
|
|
|
//
|
2009-01-29 05:54:33 +08:00
|
|
|
// struct X { int a, b; };
|
|
|
|
// struct X xs[] = { [0] = (struct X) { 1, 2 }, [0].b = 3 };
|
2009-09-09 23:08:12 +08:00
|
|
|
//
|
2009-01-29 05:54:33 +08:00
|
|
|
// Here, xs[0].a == 0 and xs[0].b == 3, since the second,
|
|
|
|
// designated initializer re-initializes the whole
|
|
|
|
// subobject [0], overwriting previous initializers.
|
2009-09-09 23:08:12 +08:00
|
|
|
SemaRef.Diag(InitRange.getBegin(),
|
2009-03-02 01:12:46 +08:00
|
|
|
diag::warn_subobject_initializer_overrides)
|
2009-01-29 05:54:33 +08:00
|
|
|
<< InitRange;
|
2012-03-10 02:35:03 +08:00
|
|
|
SemaRef.Diag(ExistingInit->getLocStart(),
|
2009-01-29 05:54:33 +08:00
|
|
|
diag::note_previous_initializer)
|
2009-01-29 07:43:32 +08:00
|
|
|
<< /*FIXME:has side effects=*/0
|
2009-01-29 05:54:33 +08:00
|
|
|
<< ExistingInit->getSourceRange();
|
|
|
|
}
|
|
|
|
|
2009-09-09 23:08:12 +08:00
|
|
|
InitListExpr *Result
|
2010-04-14 07:39:13 +08:00
|
|
|
= new (SemaRef.Context) InitListExpr(SemaRef.Context,
|
2013-05-06 04:40:26 +08:00
|
|
|
InitRange.getBegin(), None,
|
2010-02-19 09:50:18 +08:00
|
|
|
InitRange.getEnd());
|
2009-03-02 01:12:46 +08:00
|
|
|
|
2012-02-23 10:25:10 +08:00
|
|
|
QualType ResultType = CurrentObjectType;
|
|
|
|
if (!ResultType->isArrayType())
|
|
|
|
ResultType = ResultType.getNonLValueExprType(SemaRef.Context);
|
|
|
|
Result->setType(ResultType);
|
2009-01-29 05:54:33 +08:00
|
|
|
|
2009-03-21 07:58:33 +08:00
|
|
|
// Pre-allocate storage for the structured initializer list.
|
|
|
|
unsigned NumElements = 0;
|
2009-03-22 02:13:52 +08:00
|
|
|
unsigned NumInits = 0;
|
2011-04-29 02:53:55 +08:00
|
|
|
bool GotNumInits = false;
|
|
|
|
if (!StructuredList) {
|
2009-03-22 02:13:52 +08:00
|
|
|
NumInits = IList->getNumInits();
|
2011-04-29 02:53:55 +08:00
|
|
|
GotNumInits = true;
|
|
|
|
} else if (Index < IList->getNumInits()) {
|
|
|
|
if (InitListExpr *SubList = dyn_cast<InitListExpr>(IList->getInit(Index))) {
|
2009-03-22 02:13:52 +08:00
|
|
|
NumInits = SubList->getNumInits();
|
2011-04-29 02:53:55 +08:00
|
|
|
GotNumInits = true;
|
|
|
|
}
|
2009-03-22 02:13:52 +08:00
|
|
|
}
|
|
|
|
|
2009-09-09 23:08:12 +08:00
|
|
|
if (const ArrayType *AType
|
2009-03-21 07:58:33 +08:00
|
|
|
= SemaRef.Context.getAsArrayType(CurrentObjectType)) {
|
|
|
|
if (const ConstantArrayType *CAType = dyn_cast<ConstantArrayType>(AType)) {
|
|
|
|
NumElements = CAType->getSize().getZExtValue();
|
|
|
|
// Simple heuristic so that we don't allocate a very large
|
|
|
|
// initializer with many empty entries at the end.
|
2011-04-29 02:53:55 +08:00
|
|
|
if (GotNumInits && NumElements > NumInits)
|
2009-03-21 07:58:33 +08:00
|
|
|
NumElements = 0;
|
|
|
|
}
|
2009-09-22 07:43:11 +08:00
|
|
|
} else if (const VectorType *VType = CurrentObjectType->getAs<VectorType>())
|
2009-03-21 07:58:33 +08:00
|
|
|
NumElements = VType->getNumElements();
|
2009-07-30 05:53:49 +08:00
|
|
|
else if (const RecordType *RType = CurrentObjectType->getAs<RecordType>()) {
|
2009-03-21 07:58:33 +08:00
|
|
|
RecordDecl *RDecl = RType->getDecl();
|
|
|
|
if (RDecl->isUnion())
|
|
|
|
NumElements = 1;
|
|
|
|
else
|
2009-09-09 23:08:12 +08:00
|
|
|
NumElements = std::distance(RDecl->field_begin(),
|
2009-06-30 10:36:12 +08:00
|
|
|
RDecl->field_end());
|
2009-03-21 07:58:33 +08:00
|
|
|
}
|
|
|
|
|
2010-04-14 07:39:13 +08:00
|
|
|
Result->reserveInits(SemaRef.Context, NumElements);
|
2009-03-21 07:58:33 +08:00
|
|
|
|
2009-01-29 05:54:33 +08:00
|
|
|
// Link this new initializer list into the structured initializer
|
|
|
|
// lists.
|
|
|
|
if (StructuredList)
|
2010-04-14 07:39:13 +08:00
|
|
|
StructuredList->updateInit(SemaRef.Context, StructuredIndex, Result);
|
2009-01-29 05:54:33 +08:00
|
|
|
else {
|
|
|
|
Result->setSyntacticForm(IList);
|
|
|
|
SyntacticToSemantic[IList] = Result;
|
|
|
|
}
|
|
|
|
|
|
|
|
return Result;
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Update the initializer at index @p StructuredIndex within the
|
|
|
|
/// structured initializer list to the value @p expr.
|
|
|
|
void InitListChecker::UpdateStructuredListElement(InitListExpr *StructuredList,
|
|
|
|
unsigned &StructuredIndex,
|
|
|
|
Expr *expr) {
|
|
|
|
// No structured initializer list to update
|
|
|
|
if (!StructuredList)
|
|
|
|
return;
|
|
|
|
|
2010-04-14 07:39:13 +08:00
|
|
|
if (Expr *PrevInit = StructuredList->updateInit(SemaRef.Context,
|
|
|
|
StructuredIndex, expr)) {
|
2009-01-29 05:54:33 +08:00
|
|
|
// This initializer overwrites a previous initializer. Warn.
|
2012-03-10 02:35:03 +08:00
|
|
|
SemaRef.Diag(expr->getLocStart(),
|
2009-01-29 05:54:33 +08:00
|
|
|
diag::warn_initializer_overrides)
|
|
|
|
<< expr->getSourceRange();
|
2012-03-10 02:35:03 +08:00
|
|
|
SemaRef.Diag(PrevInit->getLocStart(),
|
2009-01-29 05:54:33 +08:00
|
|
|
diag::note_previous_initializer)
|
2009-01-29 07:43:32 +08:00
|
|
|
<< /*FIXME:has side effects=*/0
|
2009-01-29 05:54:33 +08:00
|
|
|
<< PrevInit->getSourceRange();
|
|
|
|
}
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-01-29 05:54:33 +08:00
|
|
|
++StructuredIndex;
|
|
|
|
}
|
|
|
|
|
2009-01-22 08:58:24 +08:00
|
|
|
/// Check that the given Index expression is a valid array designator
|
2012-02-04 17:53:13 +08:00
|
|
|
/// value. This is essentially just a wrapper around
|
2009-04-26 05:59:05 +08:00
|
|
|
/// VerifyIntegerConstantExpression that also checks for negative values
|
2009-01-22 08:58:24 +08:00
|
|
|
/// and produces a reasonable diagnostic if there is a
|
2012-02-04 17:53:13 +08:00
|
|
|
/// failure. Returns the index expression, possibly with an implicit cast
|
|
|
|
/// added, on success. If everything went okay, Value will receive the
|
|
|
|
/// value of the constant expression.
|
|
|
|
static ExprResult
|
2009-04-26 05:59:05 +08:00
|
|
|
CheckArrayDesignatorExpr(Sema &S, Expr *Index, llvm::APSInt &Value) {
|
2012-03-10 02:35:03 +08:00
|
|
|
SourceLocation Loc = Index->getLocStart();
|
2009-01-22 08:58:24 +08:00
|
|
|
|
|
|
|
// Make sure this is an integer constant expression.
|
2012-02-04 17:53:13 +08:00
|
|
|
ExprResult Result = S.VerifyIntegerConstantExpression(Index, &Value);
|
|
|
|
if (Result.isInvalid())
|
|
|
|
return Result;
|
2009-04-26 05:59:05 +08:00
|
|
|
|
|
|
|
if (Value.isSigned() && Value.isNegative())
|
|
|
|
return S.Diag(Loc, diag::err_array_designator_negative)
|
2009-01-22 08:58:24 +08:00
|
|
|
<< Value.toString(10) << Index->getSourceRange();
|
|
|
|
|
2009-01-24 05:04:18 +08:00
|
|
|
Value.setIsUnsigned(true);
|
2012-02-04 17:53:13 +08:00
|
|
|
return Result;
|
2009-01-22 08:58:24 +08:00
|
|
|
}
|
|
|
|
|
2010-08-24 14:29:42 +08:00
|
|
|
ExprResult Sema::ActOnDesignatedInitializer(Designation &Desig,
|
2010-11-20 09:29:55 +08:00
|
|
|
SourceLocation Loc,
|
|
|
|
bool GNUSyntax,
|
|
|
|
ExprResult Init) {
|
2009-01-22 08:58:24 +08:00
|
|
|
typedef DesignatedInitExpr::Designator ASTDesignator;
|
|
|
|
|
|
|
|
bool Invalid = false;
|
2011-07-23 18:55:15 +08:00
|
|
|
SmallVector<ASTDesignator, 32> Designators;
|
|
|
|
SmallVector<Expr *, 32> InitExpressions;
|
2009-01-22 08:58:24 +08:00
|
|
|
|
|
|
|
// Build designators and check array designator expressions.
|
|
|
|
for (unsigned Idx = 0; Idx < Desig.getNumDesignators(); ++Idx) {
|
|
|
|
const Designator &D = Desig.getDesignator(Idx);
|
|
|
|
switch (D.getKind()) {
|
|
|
|
case Designator::FieldDesignator:
|
2009-09-09 23:08:12 +08:00
|
|
|
Designators.push_back(ASTDesignator(D.getField(), D.getDotLoc(),
|
2009-01-22 08:58:24 +08:00
|
|
|
D.getFieldLoc()));
|
|
|
|
break;
|
|
|
|
|
|
|
|
case Designator::ArrayDesignator: {
|
|
|
|
Expr *Index = static_cast<Expr *>(D.getArrayIndex());
|
|
|
|
llvm::APSInt IndexValue;
|
2012-02-04 17:53:13 +08:00
|
|
|
if (!Index->isTypeDependent() && !Index->isValueDependent())
|
|
|
|
Index = CheckArrayDesignatorExpr(*this, Index, IndexValue).take();
|
|
|
|
if (!Index)
|
2009-01-22 08:58:24 +08:00
|
|
|
Invalid = true;
|
|
|
|
else {
|
|
|
|
Designators.push_back(ASTDesignator(InitExpressions.size(),
|
2009-09-09 23:08:12 +08:00
|
|
|
D.getLBracketLoc(),
|
2009-01-22 08:58:24 +08:00
|
|
|
D.getRBracketLoc()));
|
|
|
|
InitExpressions.push_back(Index);
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
case Designator::ArrayRangeDesignator: {
|
|
|
|
Expr *StartIndex = static_cast<Expr *>(D.getArrayRangeStart());
|
|
|
|
Expr *EndIndex = static_cast<Expr *>(D.getArrayRangeEnd());
|
|
|
|
llvm::APSInt StartValue;
|
|
|
|
llvm::APSInt EndValue;
|
2009-05-22 07:17:49 +08:00
|
|
|
bool StartDependent = StartIndex->isTypeDependent() ||
|
|
|
|
StartIndex->isValueDependent();
|
|
|
|
bool EndDependent = EndIndex->isTypeDependent() ||
|
|
|
|
EndIndex->isValueDependent();
|
2012-02-04 17:53:13 +08:00
|
|
|
if (!StartDependent)
|
|
|
|
StartIndex =
|
|
|
|
CheckArrayDesignatorExpr(*this, StartIndex, StartValue).take();
|
|
|
|
if (!EndDependent)
|
|
|
|
EndIndex = CheckArrayDesignatorExpr(*this, EndIndex, EndValue).take();
|
|
|
|
|
|
|
|
if (!StartIndex || !EndIndex)
|
2009-01-22 08:58:24 +08:00
|
|
|
Invalid = true;
|
2009-01-24 06:22:29 +08:00
|
|
|
else {
|
|
|
|
// Make sure we're comparing values with the same bit width.
|
2009-05-22 07:17:49 +08:00
|
|
|
if (StartDependent || EndDependent) {
|
|
|
|
// Nothing to compute.
|
|
|
|
} else if (StartValue.getBitWidth() > EndValue.getBitWidth())
|
2010-12-07 16:25:34 +08:00
|
|
|
EndValue = EndValue.extend(StartValue.getBitWidth());
|
2009-01-24 06:22:29 +08:00
|
|
|
else if (StartValue.getBitWidth() < EndValue.getBitWidth())
|
2010-12-07 16:25:34 +08:00
|
|
|
StartValue = StartValue.extend(EndValue.getBitWidth());
|
2009-01-24 06:22:29 +08:00
|
|
|
|
2009-05-22 07:30:39 +08:00
|
|
|
if (!StartDependent && !EndDependent && EndValue < StartValue) {
|
2009-01-24 06:22:29 +08:00
|
|
|
Diag(D.getEllipsisLoc(), diag::err_array_designator_empty_range)
|
2009-09-09 23:08:12 +08:00
|
|
|
<< StartValue.toString(10) << EndValue.toString(10)
|
2009-01-24 06:22:29 +08:00
|
|
|
<< StartIndex->getSourceRange() << EndIndex->getSourceRange();
|
|
|
|
Invalid = true;
|
|
|
|
} else {
|
|
|
|
Designators.push_back(ASTDesignator(InitExpressions.size(),
|
2009-09-09 23:08:12 +08:00
|
|
|
D.getLBracketLoc(),
|
2009-01-24 06:22:29 +08:00
|
|
|
D.getEllipsisLoc(),
|
|
|
|
D.getRBracketLoc()));
|
|
|
|
InitExpressions.push_back(StartIndex);
|
|
|
|
InitExpressions.push_back(EndIndex);
|
|
|
|
}
|
2009-01-22 08:58:24 +08:00
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (Invalid || Init.isInvalid())
|
|
|
|
return ExprError();
|
|
|
|
|
|
|
|
// Clear out the expressions within the designation.
|
|
|
|
Desig.ClearExprs(*this);
|
|
|
|
|
|
|
|
DesignatedInitExpr *DIE
|
2009-05-21 17:52:38 +08:00
|
|
|
= DesignatedInitExpr::Create(Context,
|
|
|
|
Designators.data(), Designators.size(),
|
2012-08-24 19:54:20 +08:00
|
|
|
InitExpressions, Loc, GNUSyntax,
|
|
|
|
Init.takeAs<Expr>());
|
2011-01-27 15:10:08 +08:00
|
|
|
|
2012-03-11 15:00:24 +08:00
|
|
|
if (!getLangOpts().C99)
|
2011-01-17 00:13:16 +08:00
|
|
|
Diag(DIE->getLocStart(), diag::ext_designated_init)
|
|
|
|
<< DIE->getSourceRange();
|
2011-01-27 15:10:08 +08:00
|
|
|
|
2009-01-22 08:58:24 +08:00
|
|
|
return Owned(DIE);
|
|
|
|
}
|
2009-01-29 08:45:39 +08:00
|
|
|
|
2009-12-16 14:35:08 +08:00
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
// Initialization entity
|
|
|
|
//===----------------------------------------------------------------------===//
|
2009-09-10 07:08:42 +08:00
|
|
|
|
2011-01-27 15:10:08 +08:00
|
|
|
InitializedEntity::InitializedEntity(ASTContext &Context, unsigned Index,
|
2009-12-16 14:35:08 +08:00
|
|
|
const InitializedEntity &Parent)
|
2011-01-27 15:10:08 +08:00
|
|
|
: Parent(&Parent), Index(Index)
|
2009-12-16 14:35:08 +08:00
|
|
|
{
|
2010-01-23 12:34:47 +08:00
|
|
|
if (const ArrayType *AT = Context.getAsArrayType(Parent.getType())) {
|
|
|
|
Kind = EK_ArrayElement;
|
2009-12-22 23:35:07 +08:00
|
|
|
Type = AT->getElementType();
|
2011-09-20 07:17:44 +08:00
|
|
|
} else if (const VectorType *VT = Parent.getType()->getAs<VectorType>()) {
|
2010-01-23 12:34:47 +08:00
|
|
|
Kind = EK_VectorElement;
|
2011-09-20 07:17:44 +08:00
|
|
|
Type = VT->getElementType();
|
|
|
|
} else {
|
|
|
|
const ComplexType *CT = Parent.getType()->getAs<ComplexType>();
|
|
|
|
assert(CT && "Unexpected type");
|
|
|
|
Kind = EK_ComplexElement;
|
|
|
|
Type = CT->getElementType();
|
2010-01-23 12:34:47 +08:00
|
|
|
}
|
2009-12-10 07:02:17 +08:00
|
|
|
}
|
|
|
|
|
2013-07-24 23:28:33 +08:00
|
|
|
InitializedEntity
|
|
|
|
InitializedEntity::InitializeBase(ASTContext &Context,
|
|
|
|
const CXXBaseSpecifier *Base,
|
|
|
|
bool IsInheritedVirtualBase) {
|
2009-12-10 07:02:17 +08:00
|
|
|
InitializedEntity Result;
|
|
|
|
Result.Kind = EK_Base;
|
2013-06-13 05:51:50 +08:00
|
|
|
Result.Parent = 0;
|
2010-04-22 03:52:01 +08:00
|
|
|
Result.Base = reinterpret_cast<uintptr_t>(Base);
|
|
|
|
if (IsInheritedVirtualBase)
|
|
|
|
Result.Base |= 0x01;
|
2011-01-27 15:10:08 +08:00
|
|
|
|
2009-12-22 23:35:07 +08:00
|
|
|
Result.Type = Base->getType();
|
2009-12-10 07:02:17 +08:00
|
|
|
return Result;
|
|
|
|
}
|
|
|
|
|
2009-12-16 09:38:02 +08:00
|
|
|
DeclarationName InitializedEntity::getName() const {
|
|
|
|
switch (getKind()) {
|
2013-08-01 02:21:45 +08:00
|
|
|
case EK_Parameter:
|
|
|
|
case EK_Parameter_CF_Audited: {
|
2011-06-16 07:02:42 +08:00
|
|
|
ParmVarDecl *D = reinterpret_cast<ParmVarDecl*>(Parameter & ~0x1);
|
|
|
|
return (D ? D->getDeclName() : DeclarationName());
|
|
|
|
}
|
2009-12-23 00:09:06 +08:00
|
|
|
|
|
|
|
case EK_Variable:
|
2009-12-16 09:38:02 +08:00
|
|
|
case EK_Member:
|
|
|
|
return VariableOrMember->getDeclName();
|
|
|
|
|
2012-02-16 00:57:26 +08:00
|
|
|
case EK_LambdaCapture:
|
2013-12-05 09:40:41 +08:00
|
|
|
return DeclarationName(Capture.VarID);
|
2012-02-16 00:57:26 +08:00
|
|
|
|
2009-12-16 09:38:02 +08:00
|
|
|
case EK_Result:
|
|
|
|
case EK_Exception:
|
Switch the initialization required by return statements over to the
new InitializationSequence. This fixes some bugs (e.g., PR5808),
changed some diagnostics, and caused more churn than expected. What's
new:
- InitializationSequence now has a "C conversion sequence" category
and step kind, which falls back to
- Changed the diagnostics for returns to always have the result type
of the function first and the type of the expression second.
CheckSingleAssignmentConstraints to peform checking in C.
- Improved ASTs for initialization of return values. The ASTs now
capture all of the temporaries we need to create, but
intentionally do not bind the tempoary that is actually returned,
so that it won't get destroyed twice.
- Make sure to perform an (elidable!) copy of the class object that
is returned from a class.
- Fix copy elision in CodeGen to properly see through the
subexpressions that occur with elidable copies.
- Give "new" its own entity kind; as with return values and thrown
objects, we don't bind the expression so we don't call a
destructor for it.
Note that, with this patch, I've broken returning move-only types in
C++0x. We'll fix it later, when we tackle NRVO.
llvm-svn: 91669
2009-12-18 13:02:21 +08:00
|
|
|
case EK_New:
|
2009-12-16 09:38:02 +08:00
|
|
|
case EK_Temporary:
|
|
|
|
case EK_Base:
|
2011-05-01 15:04:31 +08:00
|
|
|
case EK_Delegating:
|
2010-01-23 12:34:47 +08:00
|
|
|
case EK_ArrayElement:
|
|
|
|
case EK_VectorElement:
|
2011-09-20 07:17:44 +08:00
|
|
|
case EK_ComplexElement:
|
2010-06-08 00:14:00 +08:00
|
|
|
case EK_BlockElement:
|
2013-05-07 00:48:12 +08:00
|
|
|
case EK_CompoundLiteralInit:
|
2013-07-12 00:48:06 +08:00
|
|
|
case EK_RelatedResult:
|
2009-12-16 09:38:02 +08:00
|
|
|
return DeclarationName();
|
|
|
|
}
|
2011-01-27 15:10:08 +08:00
|
|
|
|
2012-01-17 14:56:22 +08:00
|
|
|
llvm_unreachable("Invalid EntityKind!");
|
2009-12-16 09:38:02 +08:00
|
|
|
}
|
|
|
|
|
2009-12-19 11:01:41 +08:00
|
|
|
DeclaratorDecl *InitializedEntity::getDecl() const {
|
|
|
|
switch (getKind()) {
|
|
|
|
case EK_Variable:
|
|
|
|
case EK_Member:
|
|
|
|
return VariableOrMember;
|
|
|
|
|
2011-06-16 07:02:42 +08:00
|
|
|
case EK_Parameter:
|
2013-08-01 02:21:45 +08:00
|
|
|
case EK_Parameter_CF_Audited:
|
2011-06-16 07:02:42 +08:00
|
|
|
return reinterpret_cast<ParmVarDecl*>(Parameter & ~0x1);
|
|
|
|
|
2009-12-19 11:01:41 +08:00
|
|
|
case EK_Result:
|
|
|
|
case EK_Exception:
|
|
|
|
case EK_New:
|
|
|
|
case EK_Temporary:
|
|
|
|
case EK_Base:
|
2011-05-01 15:04:31 +08:00
|
|
|
case EK_Delegating:
|
2010-01-23 12:34:47 +08:00
|
|
|
case EK_ArrayElement:
|
|
|
|
case EK_VectorElement:
|
2011-09-20 07:17:44 +08:00
|
|
|
case EK_ComplexElement:
|
2010-06-08 00:14:00 +08:00
|
|
|
case EK_BlockElement:
|
2012-02-16 00:57:26 +08:00
|
|
|
case EK_LambdaCapture:
|
2013-05-07 00:48:12 +08:00
|
|
|
case EK_CompoundLiteralInit:
|
2013-07-12 00:48:06 +08:00
|
|
|
case EK_RelatedResult:
|
2009-12-19 11:01:41 +08:00
|
|
|
return 0;
|
|
|
|
}
|
2011-01-27 15:10:08 +08:00
|
|
|
|
2012-01-17 14:56:22 +08:00
|
|
|
llvm_unreachable("Invalid EntityKind!");
|
2009-12-19 11:01:41 +08:00
|
|
|
}
|
|
|
|
|
2010-05-15 08:13:29 +08:00
|
|
|
bool InitializedEntity::allowsNRVO() const {
|
|
|
|
switch (getKind()) {
|
|
|
|
case EK_Result:
|
|
|
|
case EK_Exception:
|
|
|
|
return LocAndNRVO.NRVO;
|
2011-01-27 15:10:08 +08:00
|
|
|
|
2010-05-15 08:13:29 +08:00
|
|
|
case EK_Variable:
|
|
|
|
case EK_Parameter:
|
2013-08-01 02:21:45 +08:00
|
|
|
case EK_Parameter_CF_Audited:
|
2010-05-15 08:13:29 +08:00
|
|
|
case EK_Member:
|
|
|
|
case EK_New:
|
|
|
|
case EK_Temporary:
|
2013-05-07 00:48:12 +08:00
|
|
|
case EK_CompoundLiteralInit:
|
2010-05-15 08:13:29 +08:00
|
|
|
case EK_Base:
|
2011-05-01 15:04:31 +08:00
|
|
|
case EK_Delegating:
|
2010-05-15 08:13:29 +08:00
|
|
|
case EK_ArrayElement:
|
|
|
|
case EK_VectorElement:
|
2011-09-20 07:17:44 +08:00
|
|
|
case EK_ComplexElement:
|
2010-06-08 00:14:00 +08:00
|
|
|
case EK_BlockElement:
|
2012-02-16 00:57:26 +08:00
|
|
|
case EK_LambdaCapture:
|
2013-07-12 00:48:06 +08:00
|
|
|
case EK_RelatedResult:
|
2010-05-15 08:13:29 +08:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2013-06-05 08:46:14 +08:00
|
|
|
unsigned InitializedEntity::dumpImpl(raw_ostream &OS) const {
|
2013-06-13 05:51:50 +08:00
|
|
|
assert(getParent() != this);
|
2013-06-05 08:46:14 +08:00
|
|
|
unsigned Depth = getParent() ? getParent()->dumpImpl(OS) : 0;
|
|
|
|
for (unsigned I = 0; I != Depth; ++I)
|
|
|
|
OS << "`-";
|
|
|
|
|
|
|
|
switch (getKind()) {
|
|
|
|
case EK_Variable: OS << "Variable"; break;
|
|
|
|
case EK_Parameter: OS << "Parameter"; break;
|
2013-08-01 02:21:45 +08:00
|
|
|
case EK_Parameter_CF_Audited: OS << "CF audited function Parameter";
|
|
|
|
break;
|
2013-06-05 08:46:14 +08:00
|
|
|
case EK_Result: OS << "Result"; break;
|
|
|
|
case EK_Exception: OS << "Exception"; break;
|
|
|
|
case EK_Member: OS << "Member"; break;
|
|
|
|
case EK_New: OS << "New"; break;
|
|
|
|
case EK_Temporary: OS << "Temporary"; break;
|
|
|
|
case EK_CompoundLiteralInit: OS << "CompoundLiteral";break;
|
2013-07-12 00:48:06 +08:00
|
|
|
case EK_RelatedResult: OS << "RelatedResult"; break;
|
2013-06-05 08:46:14 +08:00
|
|
|
case EK_Base: OS << "Base"; break;
|
|
|
|
case EK_Delegating: OS << "Delegating"; break;
|
|
|
|
case EK_ArrayElement: OS << "ArrayElement " << Index; break;
|
|
|
|
case EK_VectorElement: OS << "VectorElement " << Index; break;
|
|
|
|
case EK_ComplexElement: OS << "ComplexElement " << Index; break;
|
|
|
|
case EK_BlockElement: OS << "Block"; break;
|
|
|
|
case EK_LambdaCapture:
|
|
|
|
OS << "LambdaCapture ";
|
2013-12-05 09:40:41 +08:00
|
|
|
OS << DeclarationName(Capture.VarID);
|
2013-06-05 08:46:14 +08:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (Decl *D = getDecl()) {
|
|
|
|
OS << " ";
|
|
|
|
cast<NamedDecl>(D)->printQualifiedName(OS);
|
|
|
|
}
|
|
|
|
|
|
|
|
OS << " '" << getType().getAsString() << "'\n";
|
|
|
|
|
|
|
|
return Depth + 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
void InitializedEntity::dump() const {
|
|
|
|
dumpImpl(llvm::errs());
|
|
|
|
}
|
|
|
|
|
2009-12-10 07:02:17 +08:00
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
// Initialization sequence
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
|
|
|
|
void InitializationSequence::Step::Destroy() {
|
|
|
|
switch (Kind) {
|
|
|
|
case SK_ResolveAddressOfOverloadedFunction:
|
|
|
|
case SK_CastDerivedToBaseRValue:
|
2010-07-20 12:20:21 +08:00
|
|
|
case SK_CastDerivedToBaseXValue:
|
2009-12-10 07:02:17 +08:00
|
|
|
case SK_CastDerivedToBaseLValue:
|
|
|
|
case SK_BindReference:
|
|
|
|
case SK_BindReferenceToTemporary:
|
2010-04-18 15:40:54 +08:00
|
|
|
case SK_ExtraneousCopyToTemporary:
|
2009-12-10 07:02:17 +08:00
|
|
|
case SK_UserConversion:
|
|
|
|
case SK_QualificationConversionRValue:
|
2010-07-20 12:20:21 +08:00
|
|
|
case SK_QualificationConversionXValue:
|
2009-12-10 07:02:17 +08:00
|
|
|
case SK_QualificationConversionLValue:
|
2013-04-11 08:58:58 +08:00
|
|
|
case SK_LValueToRValue:
|
2009-12-11 01:56:55 +08:00
|
|
|
case SK_ListInitialization:
|
2011-09-25 01:47:52 +08:00
|
|
|
case SK_ListConstructorCall:
|
2011-11-28 00:50:07 +08:00
|
|
|
case SK_UnwrapInitList:
|
|
|
|
case SK_RewrapInitList:
|
2009-12-15 04:49:26 +08:00
|
|
|
case SK_ConstructorInitialization:
|
2009-12-15 08:01:57 +08:00
|
|
|
case SK_ZeroInitialization:
|
Switch the initialization required by return statements over to the
new InitializationSequence. This fixes some bugs (e.g., PR5808),
changed some diagnostics, and caused more churn than expected. What's
new:
- InitializationSequence now has a "C conversion sequence" category
and step kind, which falls back to
- Changed the diagnostics for returns to always have the result type
of the function first and the type of the expression second.
CheckSingleAssignmentConstraints to peform checking in C.
- Improved ASTs for initialization of return values. The ASTs now
capture all of the temporaries we need to create, but
intentionally do not bind the tempoary that is actually returned,
so that it won't get destroyed twice.
- Make sure to perform an (elidable!) copy of the class object that
is returned from a class.
- Fix copy elision in CodeGen to properly see through the
subexpressions that occur with elidable copies.
- Give "new" its own entity kind; as with return values and thrown
objects, we don't bind the expression so we don't call a
destructor for it.
Note that, with this patch, I've broken returning move-only types in
C++0x. We'll fix it later, when we tackle NRVO.
llvm-svn: 91669
2009-12-18 13:02:21 +08:00
|
|
|
case SK_CAssignment:
|
2009-12-19 16:11:05 +08:00
|
|
|
case SK_StringInit:
|
2010-08-07 19:51:51 +08:00
|
|
|
case SK_ObjCObjectConversion:
|
2011-02-23 02:29:51 +08:00
|
|
|
case SK_ArrayInit:
|
2012-02-16 06:38:09 +08:00
|
|
|
case SK_ParenthesizedArrayInit:
|
2011-06-16 07:02:42 +08:00
|
|
|
case SK_PassByIndirectCopyRestore:
|
|
|
|
case SK_PassByIndirectRestore:
|
|
|
|
case SK_ProduceObjCObject:
|
2012-01-18 06:49:42 +08:00
|
|
|
case SK_StdInitializerList:
|
2013-02-07 18:55:47 +08:00
|
|
|
case SK_OCLSamplerInit:
|
2013-01-20 20:31:11 +08:00
|
|
|
case SK_OCLZeroEvent:
|
2009-12-10 07:02:17 +08:00
|
|
|
break;
|
2011-01-27 15:10:08 +08:00
|
|
|
|
2009-12-10 07:02:17 +08:00
|
|
|
case SK_ConversionSequence:
|
2013-09-22 05:19:19 +08:00
|
|
|
case SK_ConversionSequenceNoNarrowing:
|
2009-12-10 07:02:17 +08:00
|
|
|
delete ICS;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2010-03-27 04:14:36 +08:00
|
|
|
bool InitializationSequence::isDirectReferenceBinding() const {
|
2011-07-15 03:07:55 +08:00
|
|
|
return !Steps.empty() && Steps.back().Kind == SK_BindReference;
|
2010-03-27 04:14:36 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
bool InitializationSequence::isAmbiguous() const {
|
2011-06-05 21:59:05 +08:00
|
|
|
if (!Failed())
|
2010-03-27 04:14:36 +08:00
|
|
|
return false;
|
2011-01-27 15:10:08 +08:00
|
|
|
|
2010-03-27 04:14:36 +08:00
|
|
|
switch (getFailureKind()) {
|
|
|
|
case FK_TooManyInitsForReference:
|
|
|
|
case FK_ArrayNeedsInitList:
|
|
|
|
case FK_ArrayNeedsInitListOrStringLiteral:
|
2013-05-15 19:03:04 +08:00
|
|
|
case FK_ArrayNeedsInitListOrWideStringLiteral:
|
|
|
|
case FK_NarrowStringIntoWideCharArray:
|
|
|
|
case FK_WideStringIntoCharArray:
|
|
|
|
case FK_IncompatWideStringIntoWideChar:
|
2010-03-27 04:14:36 +08:00
|
|
|
case FK_AddressOfOverloadFailed: // FIXME: Could do better
|
|
|
|
case FK_NonConstLValueReferenceBindingToTemporary:
|
|
|
|
case FK_NonConstLValueReferenceBindingToUnrelated:
|
|
|
|
case FK_RValueReferenceBindingToLValue:
|
|
|
|
case FK_ReferenceInitDropsQualifiers:
|
|
|
|
case FK_ReferenceInitFailed:
|
|
|
|
case FK_ConversionFailed:
|
2011-04-09 02:41:53 +08:00
|
|
|
case FK_ConversionFromPropertyFailed:
|
2010-03-27 04:14:36 +08:00
|
|
|
case FK_TooManyInitsForScalar:
|
|
|
|
case FK_ReferenceBindingToInitList:
|
|
|
|
case FK_InitListBadDestinationType:
|
|
|
|
case FK_DefaultInitOfConst:
|
2010-05-21 06:12:02 +08:00
|
|
|
case FK_Incomplete:
|
2011-02-23 02:29:51 +08:00
|
|
|
case FK_ArrayTypeMismatch:
|
|
|
|
case FK_NonConstantArrayInit:
|
2011-09-25 01:47:52 +08:00
|
|
|
case FK_ListInitializationFailed:
|
2012-01-05 08:13:19 +08:00
|
|
|
case FK_VariableLengthArrayHasInitializer:
|
2011-10-18 02:40:02 +08:00
|
|
|
case FK_PlaceholderType:
|
2012-04-02 03:54:59 +08:00
|
|
|
case FK_ExplicitConstructor:
|
2010-03-27 04:14:36 +08:00
|
|
|
return false;
|
2011-01-27 15:10:08 +08:00
|
|
|
|
2010-03-27 04:14:36 +08:00
|
|
|
case FK_ReferenceInitOverloadFailed:
|
|
|
|
case FK_UserConversionOverloadFailed:
|
|
|
|
case FK_ConstructorOverloadFailed:
|
2011-12-23 02:58:38 +08:00
|
|
|
case FK_ListConstructorOverloadFailed:
|
2010-03-27 04:14:36 +08:00
|
|
|
return FailedOverloadResult == OR_Ambiguous;
|
|
|
|
}
|
2011-01-27 15:10:08 +08:00
|
|
|
|
2012-01-17 14:56:22 +08:00
|
|
|
llvm_unreachable("Invalid EntityKind!");
|
2010-03-27 04:14:36 +08:00
|
|
|
}
|
|
|
|
|
2010-04-17 06:09:46 +08:00
|
|
|
bool InitializationSequence::isConstructorInitialization() const {
|
|
|
|
return !Steps.empty() && Steps.back().Kind == SK_ConstructorInitialization;
|
|
|
|
}
|
|
|
|
|
2011-11-19 19:44:21 +08:00
|
|
|
void
|
|
|
|
InitializationSequence
|
|
|
|
::AddAddressOverloadResolutionStep(FunctionDecl *Function,
|
|
|
|
DeclAccessPair Found,
|
|
|
|
bool HadMultipleCandidates) {
|
2009-12-10 07:02:17 +08:00
|
|
|
Step S;
|
|
|
|
S.Kind = SK_ResolveAddressOfOverloadedFunction;
|
|
|
|
S.Type = Function->getType();
|
2011-11-19 19:44:21 +08:00
|
|
|
S.Function.HadMultipleCandidates = HadMultipleCandidates;
|
2010-03-19 15:35:19 +08:00
|
|
|
S.Function.Function = Function;
|
2010-03-31 05:47:33 +08:00
|
|
|
S.Function.FoundDecl = Found;
|
2009-12-10 07:02:17 +08:00
|
|
|
Steps.push_back(S);
|
|
|
|
}
|
|
|
|
|
2011-01-27 15:10:08 +08:00
|
|
|
void InitializationSequence::AddDerivedToBaseCastStep(QualType BaseType,
|
2010-08-25 18:28:54 +08:00
|
|
|
ExprValueKind VK) {
|
2009-12-10 07:02:17 +08:00
|
|
|
Step S;
|
2010-08-25 18:28:54 +08:00
|
|
|
switch (VK) {
|
|
|
|
case VK_RValue: S.Kind = SK_CastDerivedToBaseRValue; break;
|
|
|
|
case VK_XValue: S.Kind = SK_CastDerivedToBaseXValue; break;
|
|
|
|
case VK_LValue: S.Kind = SK_CastDerivedToBaseLValue; break;
|
2010-07-20 12:20:21 +08:00
|
|
|
}
|
2009-12-10 07:02:17 +08:00
|
|
|
S.Type = BaseType;
|
|
|
|
Steps.push_back(S);
|
|
|
|
}
|
|
|
|
|
2011-01-27 15:10:08 +08:00
|
|
|
void InitializationSequence::AddReferenceBindingStep(QualType T,
|
2009-12-10 07:02:17 +08:00
|
|
|
bool BindingTemporary) {
|
|
|
|
Step S;
|
|
|
|
S.Kind = BindingTemporary? SK_BindReferenceToTemporary : SK_BindReference;
|
|
|
|
S.Type = T;
|
|
|
|
Steps.push_back(S);
|
|
|
|
}
|
|
|
|
|
2010-04-18 15:40:54 +08:00
|
|
|
void InitializationSequence::AddExtraneousCopyToTemporary(QualType T) {
|
|
|
|
Step S;
|
|
|
|
S.Kind = SK_ExtraneousCopyToTemporary;
|
|
|
|
S.Type = T;
|
|
|
|
Steps.push_back(S);
|
|
|
|
}
|
|
|
|
|
2011-11-19 19:44:21 +08:00
|
|
|
void
|
|
|
|
InitializationSequence::AddUserConversionStep(FunctionDecl *Function,
|
|
|
|
DeclAccessPair FoundDecl,
|
|
|
|
QualType T,
|
|
|
|
bool HadMultipleCandidates) {
|
2009-12-10 07:02:17 +08:00
|
|
|
Step S;
|
|
|
|
S.Kind = SK_UserConversion;
|
2009-12-11 10:42:07 +08:00
|
|
|
S.Type = T;
|
2011-11-19 19:44:21 +08:00
|
|
|
S.Function.HadMultipleCandidates = HadMultipleCandidates;
|
2010-03-19 15:35:19 +08:00
|
|
|
S.Function.Function = Function;
|
|
|
|
S.Function.FoundDecl = FoundDecl;
|
2009-12-10 07:02:17 +08:00
|
|
|
Steps.push_back(S);
|
|
|
|
}
|
|
|
|
|
|
|
|
void InitializationSequence::AddQualificationConversionStep(QualType Ty,
|
2010-08-25 18:28:54 +08:00
|
|
|
ExprValueKind VK) {
|
2009-12-10 07:02:17 +08:00
|
|
|
Step S;
|
2010-08-27 00:36:35 +08:00
|
|
|
S.Kind = SK_QualificationConversionRValue; // work around a gcc warning
|
2010-08-25 18:28:54 +08:00
|
|
|
switch (VK) {
|
|
|
|
case VK_RValue:
|
2010-07-20 12:20:21 +08:00
|
|
|
S.Kind = SK_QualificationConversionRValue;
|
|
|
|
break;
|
2010-08-25 18:28:54 +08:00
|
|
|
case VK_XValue:
|
2010-07-20 12:20:21 +08:00
|
|
|
S.Kind = SK_QualificationConversionXValue;
|
|
|
|
break;
|
2010-08-25 18:28:54 +08:00
|
|
|
case VK_LValue:
|
2010-07-20 12:20:21 +08:00
|
|
|
S.Kind = SK_QualificationConversionLValue;
|
|
|
|
break;
|
|
|
|
}
|
2009-12-10 07:02:17 +08:00
|
|
|
S.Type = Ty;
|
|
|
|
Steps.push_back(S);
|
|
|
|
}
|
|
|
|
|
2013-04-11 08:58:58 +08:00
|
|
|
void InitializationSequence::AddLValueToRValueStep(QualType Ty) {
|
|
|
|
assert(!Ty.hasQualifiers() && "rvalues may not have qualifiers");
|
|
|
|
|
|
|
|
Step S;
|
|
|
|
S.Kind = SK_LValueToRValue;
|
|
|
|
S.Type = Ty;
|
|
|
|
Steps.push_back(S);
|
|
|
|
}
|
|
|
|
|
2009-12-10 07:02:17 +08:00
|
|
|
void InitializationSequence::AddConversionSequenceStep(
|
2013-09-22 05:19:19 +08:00
|
|
|
const ImplicitConversionSequence &ICS, QualType T,
|
|
|
|
bool TopLevelOfInitList) {
|
2009-12-10 07:02:17 +08:00
|
|
|
Step S;
|
2013-09-22 05:19:19 +08:00
|
|
|
S.Kind = TopLevelOfInitList ? SK_ConversionSequenceNoNarrowing
|
|
|
|
: SK_ConversionSequence;
|
2009-12-10 07:02:17 +08:00
|
|
|
S.Type = T;
|
|
|
|
S.ICS = new ImplicitConversionSequence(ICS);
|
|
|
|
Steps.push_back(S);
|
|
|
|
}
|
|
|
|
|
2009-12-11 01:56:55 +08:00
|
|
|
void InitializationSequence::AddListInitializationStep(QualType T) {
|
|
|
|
Step S;
|
|
|
|
S.Kind = SK_ListInitialization;
|
|
|
|
S.Type = T;
|
|
|
|
Steps.push_back(S);
|
|
|
|
}
|
|
|
|
|
2011-01-27 15:10:08 +08:00
|
|
|
void
|
2011-11-19 19:44:21 +08:00
|
|
|
InitializationSequence
|
|
|
|
::AddConstructorInitializationStep(CXXConstructorDecl *Constructor,
|
|
|
|
AccessSpecifier Access,
|
|
|
|
QualType T,
|
2011-12-22 22:44:04 +08:00
|
|
|
bool HadMultipleCandidates,
|
2012-02-05 05:27:47 +08:00
|
|
|
bool FromInitList, bool AsInitList) {
|
2009-12-15 04:49:26 +08:00
|
|
|
Step S;
|
2012-02-05 05:27:47 +08:00
|
|
|
S.Kind = FromInitList && !AsInitList ? SK_ListConstructorCall
|
|
|
|
: SK_ConstructorInitialization;
|
2009-12-15 04:49:26 +08:00
|
|
|
S.Type = T;
|
2011-11-19 19:44:21 +08:00
|
|
|
S.Function.HadMultipleCandidates = HadMultipleCandidates;
|
2010-03-19 15:35:19 +08:00
|
|
|
S.Function.Function = Constructor;
|
|
|
|
S.Function.FoundDecl = DeclAccessPair::make(Constructor, Access);
|
2009-12-15 04:49:26 +08:00
|
|
|
Steps.push_back(S);
|
|
|
|
}
|
|
|
|
|
2009-12-15 08:01:57 +08:00
|
|
|
void InitializationSequence::AddZeroInitializationStep(QualType T) {
|
|
|
|
Step S;
|
|
|
|
S.Kind = SK_ZeroInitialization;
|
|
|
|
S.Type = T;
|
|
|
|
Steps.push_back(S);
|
|
|
|
}
|
|
|
|
|
Switch the initialization required by return statements over to the
new InitializationSequence. This fixes some bugs (e.g., PR5808),
changed some diagnostics, and caused more churn than expected. What's
new:
- InitializationSequence now has a "C conversion sequence" category
and step kind, which falls back to
- Changed the diagnostics for returns to always have the result type
of the function first and the type of the expression second.
CheckSingleAssignmentConstraints to peform checking in C.
- Improved ASTs for initialization of return values. The ASTs now
capture all of the temporaries we need to create, but
intentionally do not bind the tempoary that is actually returned,
so that it won't get destroyed twice.
- Make sure to perform an (elidable!) copy of the class object that
is returned from a class.
- Fix copy elision in CodeGen to properly see through the
subexpressions that occur with elidable copies.
- Give "new" its own entity kind; as with return values and thrown
objects, we don't bind the expression so we don't call a
destructor for it.
Note that, with this patch, I've broken returning move-only types in
C++0x. We'll fix it later, when we tackle NRVO.
llvm-svn: 91669
2009-12-18 13:02:21 +08:00
|
|
|
void InitializationSequence::AddCAssignmentStep(QualType T) {
|
|
|
|
Step S;
|
|
|
|
S.Kind = SK_CAssignment;
|
|
|
|
S.Type = T;
|
|
|
|
Steps.push_back(S);
|
|
|
|
}
|
|
|
|
|
2009-12-19 16:11:05 +08:00
|
|
|
void InitializationSequence::AddStringInitStep(QualType T) {
|
|
|
|
Step S;
|
|
|
|
S.Kind = SK_StringInit;
|
|
|
|
S.Type = T;
|
|
|
|
Steps.push_back(S);
|
|
|
|
}
|
|
|
|
|
2010-08-07 19:51:51 +08:00
|
|
|
void InitializationSequence::AddObjCObjectConversionStep(QualType T) {
|
|
|
|
Step S;
|
|
|
|
S.Kind = SK_ObjCObjectConversion;
|
|
|
|
S.Type = T;
|
|
|
|
Steps.push_back(S);
|
|
|
|
}
|
|
|
|
|
2011-02-23 02:29:51 +08:00
|
|
|
void InitializationSequence::AddArrayInitStep(QualType T) {
|
|
|
|
Step S;
|
|
|
|
S.Kind = SK_ArrayInit;
|
|
|
|
S.Type = T;
|
|
|
|
Steps.push_back(S);
|
|
|
|
}
|
|
|
|
|
2012-02-16 06:38:09 +08:00
|
|
|
void InitializationSequence::AddParenthesizedArrayInitStep(QualType T) {
|
|
|
|
Step S;
|
|
|
|
S.Kind = SK_ParenthesizedArrayInit;
|
|
|
|
S.Type = T;
|
|
|
|
Steps.push_back(S);
|
|
|
|
}
|
|
|
|
|
2011-06-16 07:02:42 +08:00
|
|
|
void InitializationSequence::AddPassByIndirectCopyRestoreStep(QualType type,
|
|
|
|
bool shouldCopy) {
|
|
|
|
Step s;
|
|
|
|
s.Kind = (shouldCopy ? SK_PassByIndirectCopyRestore
|
|
|
|
: SK_PassByIndirectRestore);
|
|
|
|
s.Type = type;
|
|
|
|
Steps.push_back(s);
|
|
|
|
}
|
|
|
|
|
|
|
|
void InitializationSequence::AddProduceObjCObjectStep(QualType T) {
|
|
|
|
Step S;
|
|
|
|
S.Kind = SK_ProduceObjCObject;
|
|
|
|
S.Type = T;
|
|
|
|
Steps.push_back(S);
|
|
|
|
}
|
|
|
|
|
2012-01-18 06:49:42 +08:00
|
|
|
void InitializationSequence::AddStdInitializerListConstructionStep(QualType T) {
|
|
|
|
Step S;
|
|
|
|
S.Kind = SK_StdInitializerList;
|
|
|
|
S.Type = T;
|
|
|
|
Steps.push_back(S);
|
|
|
|
}
|
|
|
|
|
2013-02-07 18:55:47 +08:00
|
|
|
void InitializationSequence::AddOCLSamplerInitStep(QualType T) {
|
|
|
|
Step S;
|
|
|
|
S.Kind = SK_OCLSamplerInit;
|
|
|
|
S.Type = T;
|
|
|
|
Steps.push_back(S);
|
|
|
|
}
|
|
|
|
|
2013-01-20 20:31:11 +08:00
|
|
|
void InitializationSequence::AddOCLZeroEventStep(QualType T) {
|
|
|
|
Step S;
|
|
|
|
S.Kind = SK_OCLZeroEvent;
|
|
|
|
S.Type = T;
|
|
|
|
Steps.push_back(S);
|
|
|
|
}
|
|
|
|
|
2011-11-28 00:50:07 +08:00
|
|
|
void InitializationSequence::RewrapReferenceInitList(QualType T,
|
|
|
|
InitListExpr *Syntactic) {
|
|
|
|
assert(Syntactic->getNumInits() == 1 &&
|
|
|
|
"Can only rewrap trivial init lists.");
|
|
|
|
Step S;
|
|
|
|
S.Kind = SK_UnwrapInitList;
|
|
|
|
S.Type = Syntactic->getInit(0)->getType();
|
|
|
|
Steps.insert(Steps.begin(), S);
|
|
|
|
|
|
|
|
S.Kind = SK_RewrapInitList;
|
|
|
|
S.Type = T;
|
|
|
|
S.WrappingSyntacticList = Syntactic;
|
|
|
|
Steps.push_back(S);
|
|
|
|
}
|
|
|
|
|
2011-01-27 15:10:08 +08:00
|
|
|
void InitializationSequence::SetOverloadFailure(FailureKind Failure,
|
2009-12-10 07:02:17 +08:00
|
|
|
OverloadingResult Result) {
|
2011-06-05 21:59:11 +08:00
|
|
|
setSequenceKind(FailedSequence);
|
2009-12-10 07:02:17 +08:00
|
|
|
this->Failure = Failure;
|
|
|
|
this->FailedOverloadResult = Result;
|
|
|
|
}
|
|
|
|
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
// Attempt initialization
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
|
2011-06-16 07:02:42 +08:00
|
|
|
static void MaybeProduceObjCObject(Sema &S,
|
|
|
|
InitializationSequence &Sequence,
|
|
|
|
const InitializedEntity &Entity) {
|
2012-03-11 15:00:24 +08:00
|
|
|
if (!S.getLangOpts().ObjCAutoRefCount) return;
|
2011-06-16 07:02:42 +08:00
|
|
|
|
|
|
|
/// When initializing a parameter, produce the value if it's marked
|
|
|
|
/// __attribute__((ns_consumed)).
|
2013-08-01 02:21:45 +08:00
|
|
|
if (Entity.isParameterKind()) {
|
2011-06-16 07:02:42 +08:00
|
|
|
if (!Entity.isParameterConsumed())
|
|
|
|
return;
|
|
|
|
|
|
|
|
assert(Entity.getType()->isObjCRetainableType() &&
|
|
|
|
"consuming an object of unretainable type?");
|
|
|
|
Sequence.AddProduceObjCObjectStep(Entity.getType());
|
|
|
|
|
|
|
|
/// When initializing a return value, if the return type is a
|
|
|
|
/// retainable type, then returns need to immediately retain the
|
|
|
|
/// object. If an autorelease is required, it will be done at the
|
|
|
|
/// last instant.
|
|
|
|
} else if (Entity.getKind() == InitializedEntity::EK_Result) {
|
|
|
|
if (!Entity.getType()->isObjCRetainableType())
|
|
|
|
return;
|
|
|
|
|
|
|
|
Sequence.AddProduceObjCObjectStep(Entity.getType());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-06-13 06:31:48 +08:00
|
|
|
static void TryListInitialization(Sema &S,
|
|
|
|
const InitializedEntity &Entity,
|
|
|
|
const InitializationKind &Kind,
|
|
|
|
InitListExpr *InitList,
|
|
|
|
InitializationSequence &Sequence);
|
|
|
|
|
2012-07-05 16:39:21 +08:00
|
|
|
/// \brief When initializing from init list via constructor, handle
|
|
|
|
/// initialization of an object of type std::initializer_list<T>.
|
2011-12-22 22:44:04 +08:00
|
|
|
///
|
2012-07-05 16:39:21 +08:00
|
|
|
/// \return true if we have handled initialization of an object of type
|
|
|
|
/// std::initializer_list<T>, false otherwise.
|
|
|
|
static bool TryInitializerListConstruction(Sema &S,
|
|
|
|
InitListExpr *List,
|
|
|
|
QualType DestType,
|
|
|
|
InitializationSequence &Sequence) {
|
|
|
|
QualType E;
|
|
|
|
if (!S.isStdInitializerList(DestType, &E))
|
2012-02-15 05:14:13 +08:00
|
|
|
return false;
|
|
|
|
|
2013-06-13 06:31:48 +08:00
|
|
|
if (S.RequireCompleteType(List->getExprLoc(), E, 0)) {
|
|
|
|
Sequence.setIncompleteTypeFailure(E);
|
|
|
|
return true;
|
2011-12-22 22:44:04 +08:00
|
|
|
}
|
2013-06-13 06:31:48 +08:00
|
|
|
|
|
|
|
// Try initializing a temporary array from the init list.
|
|
|
|
QualType ArrayType = S.Context.getConstantArrayType(
|
|
|
|
E.withConst(), llvm::APInt(S.Context.getTypeSize(S.Context.getSizeType()),
|
|
|
|
List->getNumInits()),
|
|
|
|
clang::ArrayType::Normal, 0);
|
|
|
|
InitializedEntity HiddenArray =
|
|
|
|
InitializedEntity::InitializeTemporary(ArrayType);
|
|
|
|
InitializationKind Kind =
|
|
|
|
InitializationKind::CreateDirectList(List->getExprLoc());
|
|
|
|
TryListInitialization(S, HiddenArray, Kind, List, Sequence);
|
|
|
|
if (Sequence)
|
|
|
|
Sequence.AddStdInitializerListConstructionStep(DestType);
|
2012-07-05 16:39:21 +08:00
|
|
|
return true;
|
2011-12-22 22:44:04 +08:00
|
|
|
}
|
|
|
|
|
2012-02-05 05:27:39 +08:00
|
|
|
static OverloadingResult
|
|
|
|
ResolveConstructorOverload(Sema &S, SourceLocation DeclLoc,
|
2013-05-03 23:05:50 +08:00
|
|
|
MultiExprArg Args,
|
2012-02-05 05:27:39 +08:00
|
|
|
OverloadCandidateSet &CandidateSet,
|
2012-11-13 13:07:23 +08:00
|
|
|
ArrayRef<NamedDecl *> Ctors,
|
2012-02-05 05:27:39 +08:00
|
|
|
OverloadCandidateSet::iterator &Best,
|
|
|
|
bool CopyInitializing, bool AllowExplicit,
|
2012-02-29 20:47:43 +08:00
|
|
|
bool OnlyListConstructors, bool InitListSyntax) {
|
2012-02-05 05:27:39 +08:00
|
|
|
CandidateSet.clear();
|
|
|
|
|
2012-11-13 13:07:23 +08:00
|
|
|
for (ArrayRef<NamedDecl *>::iterator
|
|
|
|
Con = Ctors.begin(), ConEnd = Ctors.end(); Con != ConEnd; ++Con) {
|
2012-02-05 05:27:39 +08:00
|
|
|
NamedDecl *D = *Con;
|
|
|
|
DeclAccessPair FoundDecl = DeclAccessPair::make(D, D->getAccess());
|
|
|
|
bool SuppressUserConversions = false;
|
|
|
|
|
|
|
|
// Find the constructor (which may be a template).
|
|
|
|
CXXConstructorDecl *Constructor = 0;
|
|
|
|
FunctionTemplateDecl *ConstructorTmpl = dyn_cast<FunctionTemplateDecl>(D);
|
|
|
|
if (ConstructorTmpl)
|
|
|
|
Constructor = cast<CXXConstructorDecl>(
|
|
|
|
ConstructorTmpl->getTemplatedDecl());
|
|
|
|
else {
|
|
|
|
Constructor = cast<CXXConstructorDecl>(D);
|
|
|
|
|
2013-09-22 05:23:47 +08:00
|
|
|
// C++11 [over.best.ics]p4:
|
|
|
|
// However, when considering the argument of a constructor or
|
|
|
|
// user-defined conversion function that is a candidate:
|
|
|
|
// -- by 13.3.1.3 when invoked for the copying/moving of a temporary
|
|
|
|
// in the second step of a class copy-initialization,
|
|
|
|
// -- by 13.3.1.7 when passing the initializer list as a single
|
|
|
|
// argument or when the initializer list has exactly one elementand
|
|
|
|
// a conversion to some class X or reference to (possibly
|
|
|
|
// cv-qualified) X is considered for the first parameter of a
|
|
|
|
// constructor of X, or
|
|
|
|
// -- by 13.3.1.4, 13.3.1.5, or 13.3.1.6 in all cases,
|
|
|
|
// only standard conversion sequences and ellipsis conversion sequences
|
|
|
|
// are considered.
|
2013-05-03 23:05:50 +08:00
|
|
|
if ((CopyInitializing || (InitListSyntax && Args.size() == 1)) &&
|
2012-02-29 20:47:43 +08:00
|
|
|
Constructor->isCopyOrMoveConstructor())
|
2012-02-05 05:27:39 +08:00
|
|
|
SuppressUserConversions = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!Constructor->isInvalidDecl() &&
|
|
|
|
(AllowExplicit || !Constructor->isExplicit()) &&
|
2012-02-05 05:27:47 +08:00
|
|
|
(!OnlyListConstructors || S.isInitListConstructor(Constructor))) {
|
2012-02-05 05:27:39 +08:00
|
|
|
if (ConstructorTmpl)
|
|
|
|
S.AddTemplateOverloadCandidate(ConstructorTmpl, FoundDecl,
|
2013-05-03 23:05:50 +08:00
|
|
|
/*ExplicitArgs*/ 0, Args,
|
2012-02-29 20:47:43 +08:00
|
|
|
CandidateSet, SuppressUserConversions);
|
2012-02-25 07:56:31 +08:00
|
|
|
else {
|
|
|
|
// C++ [over.match.copy]p1:
|
|
|
|
// - When initializing a temporary to be bound to the first parameter
|
|
|
|
// of a constructor that takes a reference to possibly cv-qualified
|
|
|
|
// T as its first argument, called with a single argument in the
|
|
|
|
// context of direct-initialization, explicit conversion functions
|
|
|
|
// are also considered.
|
|
|
|
bool AllowExplicitConv = AllowExplicit && !CopyInitializing &&
|
2013-05-03 23:05:50 +08:00
|
|
|
Args.size() == 1 &&
|
2012-02-25 07:56:31 +08:00
|
|
|
Constructor->isCopyOrMoveConstructor();
|
2013-05-03 23:05:50 +08:00
|
|
|
S.AddOverloadCandidate(Constructor, FoundDecl, Args, CandidateSet,
|
2012-02-25 07:56:31 +08:00
|
|
|
SuppressUserConversions,
|
|
|
|
/*PartialOverloading=*/false,
|
|
|
|
/*AllowExplicit=*/AllowExplicitConv);
|
|
|
|
}
|
2012-02-05 05:27:39 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Perform overload resolution and return the result.
|
|
|
|
return CandidateSet.BestViableFunction(S, DeclLoc, Best);
|
|
|
|
}
|
|
|
|
|
2011-12-22 22:44:04 +08:00
|
|
|
/// \brief Attempt initialization by constructor (C++ [dcl.init]), which
|
|
|
|
/// enumerates the constructors of the initialized entity and performs overload
|
|
|
|
/// resolution to select the best.
|
2012-02-05 05:27:33 +08:00
|
|
|
/// If InitListSyntax is true, this is list-initialization of a non-aggregate
|
2011-12-22 22:44:04 +08:00
|
|
|
/// class type.
|
|
|
|
static void TryConstructorInitialization(Sema &S,
|
|
|
|
const InitializedEntity &Entity,
|
|
|
|
const InitializationKind &Kind,
|
2013-05-03 23:05:50 +08:00
|
|
|
MultiExprArg Args, QualType DestType,
|
2011-12-22 22:44:04 +08:00
|
|
|
InitializationSequence &Sequence,
|
2012-02-05 05:27:33 +08:00
|
|
|
bool InitListSyntax = false) {
|
2013-05-03 23:05:50 +08:00
|
|
|
assert((!InitListSyntax || (Args.size() == 1 && isa<InitListExpr>(Args[0]))) &&
|
2012-02-05 05:27:33 +08:00
|
|
|
"InitListSyntax must come with a single initializer list argument.");
|
|
|
|
|
2011-12-22 22:44:04 +08:00
|
|
|
// The type we're constructing needs to be complete.
|
|
|
|
if (S.RequireCompleteType(Kind.getLocation(), DestType, 0)) {
|
2012-04-11 04:43:46 +08:00
|
|
|
Sequence.setIncompleteTypeFailure(DestType);
|
2012-02-05 05:27:39 +08:00
|
|
|
return;
|
2011-12-22 22:44:04 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
const RecordType *DestRecordType = DestType->getAs<RecordType>();
|
|
|
|
assert(DestRecordType && "Constructor initialization requires record type");
|
|
|
|
CXXRecordDecl *DestRecordDecl
|
|
|
|
= cast<CXXRecordDecl>(DestRecordType->getDecl());
|
|
|
|
|
2012-02-05 05:27:39 +08:00
|
|
|
// Build the candidate set directly in the initialization sequence
|
|
|
|
// structure, so that it will persist if we fail.
|
|
|
|
OverloadCandidateSet &CandidateSet = Sequence.getFailedCandidateSet();
|
|
|
|
|
|
|
|
// Determine whether we are allowed to call explicit constructors or
|
|
|
|
// explicit conversion operators.
|
2012-04-02 03:54:59 +08:00
|
|
|
bool AllowExplicit = Kind.AllowExplicit() || InitListSyntax;
|
2012-02-05 05:27:47 +08:00
|
|
|
bool CopyInitialization = Kind.getKind() == InitializationKind::IK_Copy;
|
2012-02-05 05:27:33 +08:00
|
|
|
|
2011-12-22 22:44:04 +08:00
|
|
|
// - Otherwise, if T is a class type, constructors are considered. The
|
|
|
|
// applicable constructors are enumerated, and the best one is chosen
|
|
|
|
// through overload resolution.
|
2012-12-19 08:45:41 +08:00
|
|
|
DeclContext::lookup_result R = S.LookupConstructors(DestRecordDecl);
|
2012-11-13 13:07:23 +08:00
|
|
|
// The container holding the constructors can under certain conditions
|
|
|
|
// be changed while iterating (e.g. because of deserialization).
|
|
|
|
// To be safe we copy the lookup results to a new container.
|
2012-12-19 08:45:41 +08:00
|
|
|
SmallVector<NamedDecl*, 16> Ctors(R.begin(), R.end());
|
2011-12-22 22:44:04 +08:00
|
|
|
|
2012-02-05 05:27:47 +08:00
|
|
|
OverloadingResult Result = OR_No_Viable_Function;
|
2011-12-22 22:44:04 +08:00
|
|
|
OverloadCandidateSet::iterator Best;
|
2012-02-05 05:27:47 +08:00
|
|
|
bool AsInitializerList = false;
|
|
|
|
|
|
|
|
// C++11 [over.match.list]p1:
|
|
|
|
// When objects of non-aggregate type T are list-initialized, overload
|
|
|
|
// resolution selects the constructor in two phases:
|
|
|
|
// - Initially, the candidate functions are the initializer-list
|
|
|
|
// constructors of the class T and the argument list consists of the
|
|
|
|
// initializer list as a single argument.
|
|
|
|
if (InitListSyntax) {
|
2012-07-05 16:39:21 +08:00
|
|
|
InitListExpr *ILE = cast<InitListExpr>(Args[0]);
|
2012-02-05 05:27:47 +08:00
|
|
|
AsInitializerList = true;
|
2012-07-05 16:39:21 +08:00
|
|
|
|
|
|
|
// If the initializer list has no elements and T has a default constructor,
|
|
|
|
// the first phase is omitted.
|
2012-12-01 10:35:44 +08:00
|
|
|
if (ILE->getNumInits() != 0 || !DestRecordDecl->hasDefaultConstructor())
|
2013-05-03 23:05:50 +08:00
|
|
|
Result = ResolveConstructorOverload(S, Kind.getLocation(), Args,
|
2012-11-13 13:07:23 +08:00
|
|
|
CandidateSet, Ctors, Best,
|
2012-07-05 16:39:21 +08:00
|
|
|
CopyInitialization, AllowExplicit,
|
|
|
|
/*OnlyListConstructor=*/true,
|
|
|
|
InitListSyntax);
|
2012-02-05 05:27:47 +08:00
|
|
|
|
|
|
|
// Time to unwrap the init list.
|
2013-05-03 23:05:50 +08:00
|
|
|
Args = MultiExprArg(ILE->getInits(), ILE->getNumInits());
|
2012-02-05 05:27:47 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
// C++11 [over.match.list]p1:
|
|
|
|
// - If no viable initializer-list constructor is found, overload resolution
|
|
|
|
// is performed again, where the candidate functions are all the
|
2012-07-05 16:39:21 +08:00
|
|
|
// constructors of the class T and the argument list consists of the
|
2012-02-05 05:27:47 +08:00
|
|
|
// elements of the initializer list.
|
|
|
|
if (Result == OR_No_Viable_Function) {
|
|
|
|
AsInitializerList = false;
|
2013-05-03 23:05:50 +08:00
|
|
|
Result = ResolveConstructorOverload(S, Kind.getLocation(), Args,
|
2012-11-13 13:07:23 +08:00
|
|
|
CandidateSet, Ctors, Best,
|
2012-02-05 05:27:47 +08:00
|
|
|
CopyInitialization, AllowExplicit,
|
2012-02-29 20:47:43 +08:00
|
|
|
/*OnlyListConstructors=*/false,
|
|
|
|
InitListSyntax);
|
2012-02-05 05:27:47 +08:00
|
|
|
}
|
|
|
|
if (Result) {
|
2012-02-05 05:27:33 +08:00
|
|
|
Sequence.SetOverloadFailure(InitListSyntax ?
|
2011-12-23 02:58:38 +08:00
|
|
|
InitializationSequence::FK_ListConstructorOverloadFailed :
|
|
|
|
InitializationSequence::FK_ConstructorOverloadFailed,
|
2011-12-22 22:44:04 +08:00
|
|
|
Result);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2012-07-05 16:39:21 +08:00
|
|
|
// C++11 [dcl.init]p6:
|
2011-12-22 22:44:04 +08:00
|
|
|
// If a program calls for the default initialization of an object
|
|
|
|
// of a const-qualified type T, T shall be a class type with a
|
|
|
|
// user-provided default constructor.
|
|
|
|
if (Kind.getKind() == InitializationKind::IK_Default &&
|
|
|
|
Entity.getType().isConstQualified() &&
|
2012-08-01 06:40:31 +08:00
|
|
|
!cast<CXXConstructorDecl>(Best->Function)->isUserProvided()) {
|
2011-12-22 22:44:04 +08:00
|
|
|
Sequence.SetFailed(InitializationSequence::FK_DefaultInitOfConst);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2012-04-02 03:54:59 +08:00
|
|
|
// C++11 [over.match.list]p1:
|
|
|
|
// In copy-list-initialization, if an explicit constructor is chosen, the
|
|
|
|
// initializer is ill-formed.
|
|
|
|
CXXConstructorDecl *CtorDecl = cast<CXXConstructorDecl>(Best->Function);
|
|
|
|
if (InitListSyntax && !Kind.AllowExplicit() && CtorDecl->isExplicit()) {
|
|
|
|
Sequence.SetFailed(InitializationSequence::FK_ExplicitConstructor);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2011-12-22 22:44:04 +08:00
|
|
|
// Add the constructor initialization step. Any cv-qualification conversion is
|
|
|
|
// subsumed by the initialization.
|
|
|
|
bool HadMultipleCandidates = (CandidateSet.size() > 1);
|
|
|
|
Sequence.AddConstructorInitializationStep(CtorDecl,
|
|
|
|
Best->FoundDecl.getAccess(),
|
|
|
|
DestType, HadMultipleCandidates,
|
2012-02-05 05:27:47 +08:00
|
|
|
InitListSyntax, AsInitializerList);
|
2011-12-22 22:44:04 +08:00
|
|
|
}
|
|
|
|
|
2011-11-28 00:50:07 +08:00
|
|
|
static bool
|
|
|
|
ResolveOverloadedFunctionForReferenceBinding(Sema &S,
|
|
|
|
Expr *Initializer,
|
|
|
|
QualType &SourceType,
|
|
|
|
QualType &UnqualifiedSourceType,
|
|
|
|
QualType UnqualifiedTargetType,
|
|
|
|
InitializationSequence &Sequence) {
|
|
|
|
if (S.Context.getCanonicalType(UnqualifiedSourceType) ==
|
|
|
|
S.Context.OverloadTy) {
|
|
|
|
DeclAccessPair Found;
|
|
|
|
bool HadMultipleCandidates = false;
|
|
|
|
if (FunctionDecl *Fn
|
|
|
|
= S.ResolveAddressOfOverloadedFunction(Initializer,
|
|
|
|
UnqualifiedTargetType,
|
|
|
|
false, Found,
|
|
|
|
&HadMultipleCandidates)) {
|
|
|
|
Sequence.AddAddressOverloadResolutionStep(Fn, Found,
|
|
|
|
HadMultipleCandidates);
|
|
|
|
SourceType = Fn->getType();
|
|
|
|
UnqualifiedSourceType = SourceType.getUnqualifiedType();
|
|
|
|
} else if (!UnqualifiedTargetType->isRecordType()) {
|
|
|
|
Sequence.SetFailed(InitializationSequence::FK_AddressOfOverloadFailed);
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void TryReferenceInitializationCore(Sema &S,
|
|
|
|
const InitializedEntity &Entity,
|
|
|
|
const InitializationKind &Kind,
|
|
|
|
Expr *Initializer,
|
|
|
|
QualType cv1T1, QualType T1,
|
|
|
|
Qualifiers T1Quals,
|
|
|
|
QualType cv2T2, QualType T2,
|
|
|
|
Qualifiers T2Quals,
|
|
|
|
InitializationSequence &Sequence);
|
|
|
|
|
2012-07-05 16:39:21 +08:00
|
|
|
static void TryValueInitialization(Sema &S,
|
|
|
|
const InitializedEntity &Entity,
|
|
|
|
const InitializationKind &Kind,
|
|
|
|
InitializationSequence &Sequence,
|
|
|
|
InitListExpr *InitList = 0);
|
|
|
|
|
2011-11-28 00:50:07 +08:00
|
|
|
/// \brief Attempt list initialization of a reference.
|
|
|
|
static void TryReferenceListInitialization(Sema &S,
|
|
|
|
const InitializedEntity &Entity,
|
|
|
|
const InitializationKind &Kind,
|
|
|
|
InitListExpr *InitList,
|
2013-06-08 08:02:08 +08:00
|
|
|
InitializationSequence &Sequence) {
|
2011-11-28 00:50:07 +08:00
|
|
|
// First, catch C++03 where this isn't possible.
|
2013-01-02 19:42:31 +08:00
|
|
|
if (!S.getLangOpts().CPlusPlus11) {
|
2011-11-28 00:50:07 +08:00
|
|
|
Sequence.SetFailed(InitializationSequence::FK_ReferenceBindingToInitList);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
QualType DestType = Entity.getType();
|
|
|
|
QualType cv1T1 = DestType->getAs<ReferenceType>()->getPointeeType();
|
|
|
|
Qualifiers T1Quals;
|
|
|
|
QualType T1 = S.Context.getUnqualifiedArrayType(cv1T1, T1Quals);
|
|
|
|
|
|
|
|
// Reference initialization via an initializer list works thus:
|
|
|
|
// If the initializer list consists of a single element that is
|
|
|
|
// reference-related to the referenced type, bind directly to that element
|
|
|
|
// (possibly creating temporaries).
|
|
|
|
// Otherwise, initialize a temporary with the initializer list and
|
|
|
|
// bind to that.
|
|
|
|
if (InitList->getNumInits() == 1) {
|
|
|
|
Expr *Initializer = InitList->getInit(0);
|
|
|
|
QualType cv2T2 = Initializer->getType();
|
|
|
|
Qualifiers T2Quals;
|
|
|
|
QualType T2 = S.Context.getUnqualifiedArrayType(cv2T2, T2Quals);
|
|
|
|
|
|
|
|
// If this fails, creating a temporary wouldn't work either.
|
|
|
|
if (ResolveOverloadedFunctionForReferenceBinding(S, Initializer, cv2T2, T2,
|
|
|
|
T1, Sequence))
|
|
|
|
return;
|
|
|
|
|
|
|
|
SourceLocation DeclLoc = Initializer->getLocStart();
|
|
|
|
bool dummy1, dummy2, dummy3;
|
|
|
|
Sema::ReferenceCompareResult RefRelationship
|
|
|
|
= S.CompareReferenceRelationship(DeclLoc, cv1T1, cv2T2, dummy1,
|
|
|
|
dummy2, dummy3);
|
|
|
|
if (RefRelationship >= Sema::Ref_Related) {
|
|
|
|
// Try to bind the reference here.
|
|
|
|
TryReferenceInitializationCore(S, Entity, Kind, Initializer, cv1T1, T1,
|
|
|
|
T1Quals, cv2T2, T2, T2Quals, Sequence);
|
|
|
|
if (Sequence)
|
|
|
|
Sequence.RewrapReferenceInitList(cv1T1, InitList);
|
|
|
|
return;
|
|
|
|
}
|
2013-01-15 15:58:29 +08:00
|
|
|
|
|
|
|
// Update the initializer if we've resolved an overloaded function.
|
|
|
|
if (Sequence.step_begin() != Sequence.step_end())
|
|
|
|
Sequence.RewrapReferenceInitList(cv1T1, InitList);
|
2011-11-28 00:50:07 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
// Not reference-related. Create a temporary and bind to that.
|
|
|
|
InitializedEntity TempEntity = InitializedEntity::InitializeTemporary(cv1T1);
|
|
|
|
|
|
|
|
TryListInitialization(S, TempEntity, Kind, InitList, Sequence);
|
|
|
|
if (Sequence) {
|
|
|
|
if (DestType->isRValueReferenceType() ||
|
|
|
|
(T1Quals.hasConst() && !T1Quals.hasVolatile()))
|
|
|
|
Sequence.AddReferenceBindingStep(cv1T1, /*bindingTemporary=*/true);
|
|
|
|
else
|
|
|
|
Sequence.SetFailed(
|
|
|
|
InitializationSequence::FK_NonConstLValueReferenceBindingToTemporary);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-07-15 06:58:04 +08:00
|
|
|
/// \brief Attempt list initialization (C++0x [dcl.init.list])
|
|
|
|
static void TryListInitialization(Sema &S,
|
|
|
|
const InitializedEntity &Entity,
|
|
|
|
const InitializationKind &Kind,
|
|
|
|
InitListExpr *InitList,
|
|
|
|
InitializationSequence &Sequence) {
|
|
|
|
QualType DestType = Entity.getType();
|
|
|
|
|
2011-09-25 01:48:00 +08:00
|
|
|
// C++ doesn't allow scalar initialization with more than one argument.
|
|
|
|
// But C99 complex numbers are scalars and it makes sense there.
|
2012-03-11 15:00:24 +08:00
|
|
|
if (S.getLangOpts().CPlusPlus && DestType->isScalarType() &&
|
2011-09-25 01:48:00 +08:00
|
|
|
!DestType->isAnyComplexType() && InitList->getNumInits() > 1) {
|
|
|
|
Sequence.SetFailed(InitializationSequence::FK_TooManyInitsForScalar);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
if (DestType->isReferenceType()) {
|
2011-11-28 00:50:07 +08:00
|
|
|
TryReferenceListInitialization(S, Entity, Kind, InitList, Sequence);
|
2011-07-15 06:58:04 +08:00
|
|
|
return;
|
2011-09-25 01:48:00 +08:00
|
|
|
}
|
2012-02-19 20:27:43 +08:00
|
|
|
if (DestType->isRecordType()) {
|
2012-05-05 00:32:21 +08:00
|
|
|
if (S.RequireCompleteType(InitList->getLocStart(), DestType, 0)) {
|
2012-04-11 04:43:46 +08:00
|
|
|
Sequence.setIncompleteTypeFailure(DestType);
|
2012-02-19 20:27:43 +08:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2012-07-05 16:39:21 +08:00
|
|
|
// C++11 [dcl.init.list]p3:
|
|
|
|
// - If T is an aggregate, aggregate initialization is performed.
|
2012-02-19 20:27:43 +08:00
|
|
|
if (!DestType->isAggregateType()) {
|
2013-01-02 19:42:31 +08:00
|
|
|
if (S.getLangOpts().CPlusPlus11) {
|
2012-07-05 16:39:21 +08:00
|
|
|
// - Otherwise, if the initializer list has no elements and T is a
|
|
|
|
// class type with a default constructor, the object is
|
|
|
|
// value-initialized.
|
|
|
|
if (InitList->getNumInits() == 0) {
|
|
|
|
CXXRecordDecl *RD = DestType->getAsCXXRecordDecl();
|
2012-12-01 10:35:44 +08:00
|
|
|
if (RD->hasDefaultConstructor()) {
|
2012-07-05 16:39:21 +08:00
|
|
|
TryValueInitialization(S, Entity, Kind, Sequence, InitList);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// - Otherwise, if T is a specialization of std::initializer_list<E>,
|
|
|
|
// an initializer_list object constructed [...]
|
|
|
|
if (TryInitializerListConstruction(S, InitList, DestType, Sequence))
|
|
|
|
return;
|
|
|
|
|
|
|
|
// - Otherwise, if T is a class type, constructors are considered.
|
2013-05-03 23:05:50 +08:00
|
|
|
Expr *InitListAsExpr = InitList;
|
|
|
|
TryConstructorInitialization(S, Entity, Kind, InitListAsExpr, DestType,
|
2012-07-05 16:39:21 +08:00
|
|
|
Sequence, /*InitListSyntax*/true);
|
2012-02-19 20:27:43 +08:00
|
|
|
} else
|
|
|
|
Sequence.SetFailed(
|
|
|
|
InitializationSequence::FK_InitListBadDestinationType);
|
|
|
|
return;
|
|
|
|
}
|
2011-09-25 01:48:00 +08:00
|
|
|
}
|
2013-09-22 05:55:46 +08:00
|
|
|
if (S.getLangOpts().CPlusPlus && !DestType->isAggregateType() &&
|
|
|
|
InitList->getNumInits() == 1 &&
|
|
|
|
InitList->getInit(0)->getType()->isRecordType()) {
|
|
|
|
// - Otherwise, if the initializer list has a single element of type E
|
|
|
|
// [...references are handled above...], the object or reference is
|
|
|
|
// initialized from that element; if a narrowing conversion is required
|
|
|
|
// to convert the element to T, the program is ill-formed.
|
|
|
|
//
|
|
|
|
// Per core-24034, this is direct-initialization if we were performing
|
|
|
|
// direct-list-initialization and copy-initialization otherwise.
|
|
|
|
// We can't use InitListChecker for this, because it always performs
|
|
|
|
// copy-initialization. This only matters if we might use an 'explicit'
|
|
|
|
// conversion operator, so we only need to handle the cases where the source
|
|
|
|
// is of record type.
|
|
|
|
InitializationKind SubKind =
|
|
|
|
Kind.getKind() == InitializationKind::IK_DirectList
|
|
|
|
? InitializationKind::CreateDirect(Kind.getLocation(),
|
|
|
|
InitList->getLBraceLoc(),
|
|
|
|
InitList->getRBraceLoc())
|
|
|
|
: Kind;
|
|
|
|
Expr *SubInit[1] = { InitList->getInit(0) };
|
|
|
|
Sequence.InitializeFrom(S, Entity, SubKind, SubInit,
|
|
|
|
/*TopLevelOfInitList*/true);
|
|
|
|
if (Sequence)
|
|
|
|
Sequence.RewrapReferenceInitList(Entity.getType(), InitList);
|
|
|
|
return;
|
|
|
|
}
|
2011-09-25 01:48:00 +08:00
|
|
|
|
|
|
|
InitListChecker CheckInitList(S, Entity, InitList,
|
2013-06-06 19:41:05 +08:00
|
|
|
DestType, /*VerifyOnly=*/true);
|
2011-09-25 01:48:00 +08:00
|
|
|
if (CheckInitList.HadError()) {
|
|
|
|
Sequence.SetFailed(InitializationSequence::FK_ListInitializationFailed);
|
|
|
|
return;
|
2011-07-15 06:58:04 +08:00
|
|
|
}
|
|
|
|
|
2011-09-25 01:48:00 +08:00
|
|
|
// Add the list initialization step with the built init list.
|
2011-07-15 06:58:04 +08:00
|
|
|
Sequence.AddListInitializationStep(DestType);
|
|
|
|
}
|
2009-12-10 07:02:17 +08:00
|
|
|
|
|
|
|
/// \brief Try a reference initialization that involves calling a conversion
|
|
|
|
/// function.
|
|
|
|
static OverloadingResult TryRefInitWithConversionFunction(Sema &S,
|
|
|
|
const InitializedEntity &Entity,
|
|
|
|
const InitializationKind &Kind,
|
2012-02-25 07:56:31 +08:00
|
|
|
Expr *Initializer,
|
|
|
|
bool AllowRValues,
|
2009-12-10 07:02:17 +08:00
|
|
|
InitializationSequence &Sequence) {
|
2009-12-22 23:35:07 +08:00
|
|
|
QualType DestType = Entity.getType();
|
2009-12-10 07:02:17 +08:00
|
|
|
QualType cv1T1 = DestType->getAs<ReferenceType>()->getPointeeType();
|
|
|
|
QualType T1 = cv1T1.getUnqualifiedType();
|
|
|
|
QualType cv2T2 = Initializer->getType();
|
|
|
|
QualType T2 = cv2T2.getUnqualifiedType();
|
|
|
|
|
|
|
|
bool DerivedToBase;
|
2010-08-07 19:51:51 +08:00
|
|
|
bool ObjCConversion;
|
2011-06-16 07:02:42 +08:00
|
|
|
bool ObjCLifetimeConversion;
|
2011-01-27 15:10:08 +08:00
|
|
|
assert(!S.CompareReferenceRelationship(Initializer->getLocStart(),
|
2010-08-07 19:51:51 +08:00
|
|
|
T1, T2, DerivedToBase,
|
2011-06-16 07:02:42 +08:00
|
|
|
ObjCConversion,
|
|
|
|
ObjCLifetimeConversion) &&
|
2009-12-10 07:02:17 +08:00
|
|
|
"Must have incompatible references when binding via conversion");
|
2009-12-13 09:37:04 +08:00
|
|
|
(void)DerivedToBase;
|
2010-08-07 19:51:51 +08:00
|
|
|
(void)ObjCConversion;
|
2011-06-16 07:02:42 +08:00
|
|
|
(void)ObjCLifetimeConversion;
|
|
|
|
|
2009-12-10 07:02:17 +08:00
|
|
|
// Build the candidate set directly in the initialization sequence
|
|
|
|
// structure, so that it will persist if we fail.
|
|
|
|
OverloadCandidateSet &CandidateSet = Sequence.getFailedCandidateSet();
|
|
|
|
CandidateSet.clear();
|
|
|
|
|
|
|
|
// Determine whether we are allowed to call explicit constructors or
|
|
|
|
// explicit conversion operators.
|
2012-02-13 00:37:24 +08:00
|
|
|
bool AllowExplicit = Kind.AllowExplicit();
|
2013-09-22 05:23:47 +08:00
|
|
|
bool AllowExplicitConvs = Kind.allowExplicitConversionFunctionsInRefBinding();
|
|
|
|
|
2009-12-10 07:02:17 +08:00
|
|
|
const RecordType *T1RecordType = 0;
|
2010-05-08 03:42:26 +08:00
|
|
|
if (AllowRValues && (T1RecordType = T1->getAs<RecordType>()) &&
|
|
|
|
!S.RequireCompleteType(Kind.getLocation(), T1, 0)) {
|
2009-12-10 07:02:17 +08:00
|
|
|
// The type we're converting to is a class type. Enumerate its constructors
|
|
|
|
// to see if there is a suitable conversion.
|
|
|
|
CXXRecordDecl *T1RecordDecl = cast<CXXRecordDecl>(T1RecordType->getDecl());
|
2010-08-17 15:23:57 +08:00
|
|
|
|
2012-12-19 08:45:41 +08:00
|
|
|
DeclContext::lookup_result R = S.LookupConstructors(T1RecordDecl);
|
2012-11-13 13:07:23 +08:00
|
|
|
// The container holding the constructors can under certain conditions
|
|
|
|
// be changed while iterating (e.g. because of deserialization).
|
|
|
|
// To be safe we copy the lookup results to a new container.
|
2012-12-19 08:45:41 +08:00
|
|
|
SmallVector<NamedDecl*, 16> Ctors(R.begin(), R.end());
|
2013-07-04 11:08:24 +08:00
|
|
|
for (SmallVectorImpl<NamedDecl *>::iterator
|
2012-11-13 13:07:23 +08:00
|
|
|
CI = Ctors.begin(), CE = Ctors.end(); CI != CE; ++CI) {
|
|
|
|
NamedDecl *D = *CI;
|
2010-03-19 15:35:19 +08:00
|
|
|
DeclAccessPair FoundDecl = DeclAccessPair::make(D, D->getAccess());
|
|
|
|
|
2009-12-10 07:02:17 +08:00
|
|
|
// Find the constructor (which may be a template).
|
|
|
|
CXXConstructorDecl *Constructor = 0;
|
2010-03-19 15:35:19 +08:00
|
|
|
FunctionTemplateDecl *ConstructorTmpl = dyn_cast<FunctionTemplateDecl>(D);
|
2009-12-10 07:02:17 +08:00
|
|
|
if (ConstructorTmpl)
|
|
|
|
Constructor = cast<CXXConstructorDecl>(
|
|
|
|
ConstructorTmpl->getTemplatedDecl());
|
|
|
|
else
|
2010-03-19 15:35:19 +08:00
|
|
|
Constructor = cast<CXXConstructorDecl>(D);
|
2011-01-27 15:10:08 +08:00
|
|
|
|
2009-12-10 07:02:17 +08:00
|
|
|
if (!Constructor->isInvalidDecl() &&
|
|
|
|
Constructor->isConvertingConstructor(AllowExplicit)) {
|
|
|
|
if (ConstructorTmpl)
|
2010-03-19 15:35:19 +08:00
|
|
|
S.AddTemplateOverloadCandidate(ConstructorTmpl, FoundDecl,
|
2010-01-26 09:37:31 +08:00
|
|
|
/*ExplicitArgs*/ 0,
|
2012-02-25 19:00:22 +08:00
|
|
|
Initializer, CandidateSet,
|
2010-10-05 11:05:30 +08:00
|
|
|
/*SuppressUserConversions=*/true);
|
2009-12-10 07:02:17 +08:00
|
|
|
else
|
2010-03-19 15:35:19 +08:00
|
|
|
S.AddOverloadCandidate(Constructor, FoundDecl,
|
2012-02-25 19:00:22 +08:00
|
|
|
Initializer, CandidateSet,
|
2010-10-05 11:05:30 +08:00
|
|
|
/*SuppressUserConversions=*/true);
|
2009-12-10 07:02:17 +08:00
|
|
|
}
|
2011-01-27 15:10:08 +08:00
|
|
|
}
|
2009-12-10 07:02:17 +08:00
|
|
|
}
|
2010-08-17 15:23:57 +08:00
|
|
|
if (T1RecordType && T1RecordType->getDecl()->isInvalidDecl())
|
|
|
|
return OR_No_Viable_Function;
|
2011-01-27 15:10:08 +08:00
|
|
|
|
2010-05-08 03:42:26 +08:00
|
|
|
const RecordType *T2RecordType = 0;
|
|
|
|
if ((T2RecordType = T2->getAs<RecordType>()) &&
|
|
|
|
!S.RequireCompleteType(Kind.getLocation(), T2, 0)) {
|
2009-12-10 07:02:17 +08:00
|
|
|
// The type we're converting from is a class type, enumerate its conversion
|
|
|
|
// functions.
|
|
|
|
CXXRecordDecl *T2RecordDecl = cast<CXXRecordDecl>(T2RecordType->getDecl());
|
|
|
|
|
2012-11-28 11:56:09 +08:00
|
|
|
std::pair<CXXRecordDecl::conversion_iterator,
|
|
|
|
CXXRecordDecl::conversion_iterator>
|
|
|
|
Conversions = T2RecordDecl->getVisibleConversionFunctions();
|
|
|
|
for (CXXRecordDecl::conversion_iterator
|
|
|
|
I = Conversions.first, E = Conversions.second; I != E; ++I) {
|
2009-12-10 07:02:17 +08:00
|
|
|
NamedDecl *D = *I;
|
|
|
|
CXXRecordDecl *ActingDC = cast<CXXRecordDecl>(D->getDeclContext());
|
|
|
|
if (isa<UsingShadowDecl>(D))
|
|
|
|
D = cast<UsingShadowDecl>(D)->getTargetDecl();
|
2011-01-27 15:10:08 +08:00
|
|
|
|
2009-12-10 07:02:17 +08:00
|
|
|
FunctionTemplateDecl *ConvTemplate = dyn_cast<FunctionTemplateDecl>(D);
|
|
|
|
CXXConversionDecl *Conv;
|
|
|
|
if (ConvTemplate)
|
|
|
|
Conv = cast<CXXConversionDecl>(ConvTemplate->getTemplatedDecl());
|
|
|
|
else
|
2010-07-01 02:13:39 +08:00
|
|
|
Conv = cast<CXXConversionDecl>(D);
|
2011-01-27 15:10:08 +08:00
|
|
|
|
2009-12-10 07:02:17 +08:00
|
|
|
// If the conversion function doesn't return a reference type,
|
|
|
|
// it can't be considered for this conversion unless we're allowed to
|
|
|
|
// consider rvalues.
|
2011-01-27 15:10:08 +08:00
|
|
|
// FIXME: Do we need to make sure that we only consider conversion
|
|
|
|
// candidates with reference-compatible results? That might be needed to
|
2009-12-10 07:02:17 +08:00
|
|
|
// break recursion.
|
2012-02-25 07:56:31 +08:00
|
|
|
if ((AllowExplicitConvs || !Conv->isExplicit()) &&
|
2009-12-10 07:02:17 +08:00
|
|
|
(AllowRValues || Conv->getConversionType()->isLValueReferenceType())){
|
|
|
|
if (ConvTemplate)
|
2010-03-19 15:35:19 +08:00
|
|
|
S.AddTemplateConversionCandidate(ConvTemplate, I.getPair(),
|
2010-01-26 09:37:31 +08:00
|
|
|
ActingDC, Initializer,
|
2013-12-19 05:46:16 +08:00
|
|
|
DestType, CandidateSet,
|
|
|
|
/*AllowObjCConversionOnExplicit=*/
|
|
|
|
false);
|
2009-12-10 07:02:17 +08:00
|
|
|
else
|
2010-03-19 15:35:19 +08:00
|
|
|
S.AddConversionCandidate(Conv, I.getPair(), ActingDC,
|
2013-12-19 05:46:16 +08:00
|
|
|
Initializer, DestType, CandidateSet,
|
|
|
|
/*AllowObjCConversionOnExplicit=*/false);
|
2009-12-10 07:02:17 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2010-08-17 15:23:57 +08:00
|
|
|
if (T2RecordType && T2RecordType->getDecl()->isInvalidDecl())
|
|
|
|
return OR_No_Viable_Function;
|
2011-01-27 15:10:08 +08:00
|
|
|
|
2009-12-10 07:02:17 +08:00
|
|
|
SourceLocation DeclLoc = Initializer->getLocStart();
|
|
|
|
|
2011-01-27 15:10:08 +08:00
|
|
|
// Perform overload resolution. If it fails, return the failed result.
|
2009-12-10 07:02:17 +08:00
|
|
|
OverloadCandidateSet::iterator Best;
|
2011-01-27 15:10:08 +08:00
|
|
|
if (OverloadingResult Result
|
2010-09-12 16:07:23 +08:00
|
|
|
= CandidateSet.BestViableFunction(S, DeclLoc, Best, true))
|
2009-12-10 07:02:17 +08:00
|
|
|
return Result;
|
2009-12-11 10:42:07 +08:00
|
|
|
|
2009-12-10 07:02:17 +08:00
|
|
|
FunctionDecl *Function = Best->Function;
|
2013-02-12 16:08:54 +08:00
|
|
|
// This is the overload that will be used for this initialization step if we
|
|
|
|
// use this initialization. Mark it as referenced.
|
|
|
|
Function->setReferenced();
|
2011-02-26 03:41:05 +08:00
|
|
|
|
2009-12-11 10:42:07 +08:00
|
|
|
// Compute the returned type of the conversion.
|
2009-12-10 07:02:17 +08:00
|
|
|
if (isa<CXXConversionDecl>(Function))
|
|
|
|
T2 = Function->getResultType();
|
|
|
|
else
|
|
|
|
T2 = cv1T1;
|
2009-12-11 10:42:07 +08:00
|
|
|
|
|
|
|
// Add the user-defined conversion step.
|
2011-11-19 19:44:21 +08:00
|
|
|
bool HadMultipleCandidates = (CandidateSet.size() > 1);
|
2010-03-19 15:35:19 +08:00
|
|
|
Sequence.AddUserConversionStep(Function, Best->FoundDecl,
|
2011-11-19 19:44:21 +08:00
|
|
|
T2.getNonLValueExprType(S.Context),
|
|
|
|
HadMultipleCandidates);
|
2009-12-11 10:42:07 +08:00
|
|
|
|
2011-01-27 15:10:08 +08:00
|
|
|
// Determine whether we need to perform derived-to-base or
|
2009-12-11 10:42:07 +08:00
|
|
|
// cv-qualification adjustments.
|
2010-08-25 18:28:54 +08:00
|
|
|
ExprValueKind VK = VK_RValue;
|
2010-07-20 12:20:21 +08:00
|
|
|
if (T2->isLValueReferenceType())
|
2010-08-25 18:28:54 +08:00
|
|
|
VK = VK_LValue;
|
2010-07-20 12:20:21 +08:00
|
|
|
else if (const RValueReferenceType *RRef = T2->getAs<RValueReferenceType>())
|
2010-08-25 18:28:54 +08:00
|
|
|
VK = RRef->getPointeeType()->isFunctionType() ? VK_LValue : VK_XValue;
|
2010-07-20 12:20:21 +08:00
|
|
|
|
2009-12-10 07:02:17 +08:00
|
|
|
bool NewDerivedToBase = false;
|
2010-08-07 19:51:51 +08:00
|
|
|
bool NewObjCConversion = false;
|
2011-06-16 07:02:42 +08:00
|
|
|
bool NewObjCLifetimeConversion = false;
|
2009-12-10 07:02:17 +08:00
|
|
|
Sema::ReferenceCompareResult NewRefRelationship
|
2011-01-27 15:10:08 +08:00
|
|
|
= S.CompareReferenceRelationship(DeclLoc, T1,
|
2010-07-14 02:40:04 +08:00
|
|
|
T2.getNonLValueExprType(S.Context),
|
2011-06-16 07:02:42 +08:00
|
|
|
NewDerivedToBase, NewObjCConversion,
|
|
|
|
NewObjCLifetimeConversion);
|
2010-03-08 07:17:44 +08:00
|
|
|
if (NewRefRelationship == Sema::Ref_Incompatible) {
|
|
|
|
// If the type we've converted to is not reference-related to the
|
|
|
|
// type we're looking for, then there is another conversion step
|
|
|
|
// we need to perform to produce a temporary of the right type
|
|
|
|
// that we'll be binding to.
|
|
|
|
ImplicitConversionSequence ICS;
|
|
|
|
ICS.setStandard();
|
|
|
|
ICS.Standard = Best->FinalConversion;
|
|
|
|
T2 = ICS.Standard.getToType(2);
|
|
|
|
Sequence.AddConversionSequenceStep(ICS, T2);
|
|
|
|
} else if (NewDerivedToBase)
|
2009-12-10 07:02:17 +08:00
|
|
|
Sequence.AddDerivedToBaseCastStep(
|
|
|
|
S.Context.getQualifiedType(T1,
|
2011-01-27 15:10:08 +08:00
|
|
|
T2.getNonReferenceType().getQualifiers()),
|
2010-08-25 18:28:54 +08:00
|
|
|
VK);
|
2010-08-07 19:51:51 +08:00
|
|
|
else if (NewObjCConversion)
|
|
|
|
Sequence.AddObjCObjectConversionStep(
|
|
|
|
S.Context.getQualifiedType(T1,
|
|
|
|
T2.getNonReferenceType().getQualifiers()));
|
|
|
|
|
2009-12-10 07:02:17 +08:00
|
|
|
if (cv1T1.getQualifiers() != T2.getNonReferenceType().getQualifiers())
|
2010-08-25 18:28:54 +08:00
|
|
|
Sequence.AddQualificationConversionStep(cv1T1, VK);
|
2011-01-27 15:10:08 +08:00
|
|
|
|
2009-12-10 07:02:17 +08:00
|
|
|
Sequence.AddReferenceBindingStep(cv1T1, !T2->isReferenceType());
|
|
|
|
return OR_Success;
|
|
|
|
}
|
2011-01-27 15:10:08 +08:00
|
|
|
|
2011-10-20 00:55:56 +08:00
|
|
|
static void CheckCXX98CompatAccessibleCopy(Sema &S,
|
|
|
|
const InitializedEntity &Entity,
|
|
|
|
Expr *CurInitExpr);
|
|
|
|
|
2011-01-27 15:10:08 +08:00
|
|
|
/// \brief Attempt reference initialization (C++0x [dcl.init.ref])
|
|
|
|
static void TryReferenceInitialization(Sema &S,
|
2009-12-10 07:02:17 +08:00
|
|
|
const InitializedEntity &Entity,
|
|
|
|
const InitializationKind &Kind,
|
|
|
|
Expr *Initializer,
|
|
|
|
InitializationSequence &Sequence) {
|
2009-12-22 23:35:07 +08:00
|
|
|
QualType DestType = Entity.getType();
|
2009-12-10 07:02:17 +08:00
|
|
|
QualType cv1T1 = DestType->getAs<ReferenceType>()->getPointeeType();
|
2010-01-13 04:32:25 +08:00
|
|
|
Qualifiers T1Quals;
|
|
|
|
QualType T1 = S.Context.getUnqualifiedArrayType(cv1T1, T1Quals);
|
2009-12-10 07:02:17 +08:00
|
|
|
QualType cv2T2 = Initializer->getType();
|
2010-01-13 04:32:25 +08:00
|
|
|
Qualifiers T2Quals;
|
|
|
|
QualType T2 = S.Context.getUnqualifiedArrayType(cv2T2, T2Quals);
|
2010-07-01 02:13:39 +08:00
|
|
|
|
2009-12-10 07:02:17 +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.
|
2011-11-28 00:50:07 +08:00
|
|
|
if (ResolveOverloadedFunctionForReferenceBinding(S, Initializer, cv2T2, T2,
|
|
|
|
T1, Sequence))
|
|
|
|
return;
|
|
|
|
|
|
|
|
// Delegate everything else to a subfunction.
|
|
|
|
TryReferenceInitializationCore(S, Entity, Kind, Initializer, cv1T1, T1,
|
|
|
|
T1Quals, cv2T2, T2, T2Quals, Sequence);
|
|
|
|
}
|
2010-07-01 02:13:39 +08:00
|
|
|
|
2013-04-11 08:58:58 +08:00
|
|
|
/// Converts the target of reference initialization so that it has the
|
|
|
|
/// appropriate qualifiers and value kind.
|
|
|
|
///
|
|
|
|
/// In this case, 'x' is an 'int' lvalue, but it needs to be 'const int'.
|
|
|
|
/// \code
|
|
|
|
/// int x;
|
|
|
|
/// const int &r = x;
|
|
|
|
/// \endcode
|
|
|
|
///
|
|
|
|
/// In this case the reference is binding to a bitfield lvalue, which isn't
|
|
|
|
/// valid. Perform a load to create a lifetime-extended temporary instead.
|
|
|
|
/// \code
|
|
|
|
/// const int &r = someStruct.bitfield;
|
|
|
|
/// \endcode
|
|
|
|
static ExprValueKind
|
|
|
|
convertQualifiersAndValueKindIfNecessary(Sema &S,
|
|
|
|
InitializationSequence &Sequence,
|
|
|
|
Expr *Initializer,
|
|
|
|
QualType cv1T1,
|
|
|
|
Qualifiers T1Quals,
|
|
|
|
Qualifiers T2Quals,
|
|
|
|
bool IsLValueRef) {
|
2013-05-07 05:39:12 +08:00
|
|
|
bool IsNonAddressableType = Initializer->refersToBitField() ||
|
2013-04-11 08:58:58 +08:00
|
|
|
Initializer->refersToVectorElement();
|
|
|
|
|
|
|
|
if (IsNonAddressableType) {
|
|
|
|
// C++11 [dcl.init.ref]p5: [...] Otherwise, the reference shall be an
|
|
|
|
// lvalue reference to a non-volatile const type, or the reference shall be
|
|
|
|
// an rvalue reference.
|
|
|
|
//
|
|
|
|
// If not, we can't make a temporary and bind to that. Give up and allow the
|
|
|
|
// error to be diagnosed later.
|
|
|
|
if (IsLValueRef && (!T1Quals.hasConst() || T1Quals.hasVolatile())) {
|
|
|
|
assert(Initializer->isGLValue());
|
|
|
|
return Initializer->getValueKind();
|
|
|
|
}
|
|
|
|
|
|
|
|
// Force a load so we can materialize a temporary.
|
|
|
|
Sequence.AddLValueToRValueStep(cv1T1.getUnqualifiedType());
|
|
|
|
return VK_RValue;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (T1Quals != T2Quals) {
|
|
|
|
Sequence.AddQualificationConversionStep(cv1T1,
|
|
|
|
Initializer->getValueKind());
|
|
|
|
}
|
|
|
|
|
|
|
|
return Initializer->getValueKind();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2011-11-28 00:50:07 +08:00
|
|
|
/// \brief Reference initialization without resolving overloaded functions.
|
|
|
|
static void TryReferenceInitializationCore(Sema &S,
|
|
|
|
const InitializedEntity &Entity,
|
|
|
|
const InitializationKind &Kind,
|
|
|
|
Expr *Initializer,
|
|
|
|
QualType cv1T1, QualType T1,
|
|
|
|
Qualifiers T1Quals,
|
|
|
|
QualType cv2T2, QualType T2,
|
|
|
|
Qualifiers T2Quals,
|
|
|
|
InitializationSequence &Sequence) {
|
|
|
|
QualType DestType = Entity.getType();
|
|
|
|
SourceLocation DeclLoc = Initializer->getLocStart();
|
2009-12-10 07:02:17 +08:00
|
|
|
// Compute some basic properties of the types and the initializer.
|
|
|
|
bool isLValueRef = DestType->isLValueReferenceType();
|
|
|
|
bool isRValueRef = !isLValueRef;
|
|
|
|
bool DerivedToBase = false;
|
2010-08-07 19:51:51 +08:00
|
|
|
bool ObjCConversion = false;
|
2011-06-16 07:02:42 +08:00
|
|
|
bool ObjCLifetimeConversion = false;
|
2010-07-01 02:13:39 +08:00
|
|
|
Expr::Classification InitCategory = Initializer->Classify(S.Context);
|
2009-12-10 07:02:17 +08:00
|
|
|
Sema::ReferenceCompareResult RefRelationship
|
2010-08-07 19:51:51 +08:00
|
|
|
= S.CompareReferenceRelationship(DeclLoc, cv1T1, cv2T2, DerivedToBase,
|
2011-06-16 07:02:42 +08:00
|
|
|
ObjCConversion, ObjCLifetimeConversion);
|
2010-07-01 02:13:39 +08:00
|
|
|
|
2009-12-10 07:02:17 +08:00
|
|
|
// C++0x [dcl.init.ref]p5:
|
2011-01-27 15:10:08 +08:00
|
|
|
// A reference to type "cv1 T1" is initialized by an expression of type
|
2009-12-10 07:02:17 +08:00
|
|
|
// "cv2 T2" as follows:
|
|
|
|
//
|
2011-01-27 15:10:08 +08:00
|
|
|
// - If the reference is an lvalue reference and the initializer
|
2009-12-10 07:02:17 +08:00
|
|
|
// expression
|
2013-09-22 05:23:47 +08:00
|
|
|
// Note the analogous bullet points for rvalue refs to functions. Because
|
2010-07-01 02:13:39 +08:00
|
|
|
// there are no function rvalues in C++, rvalue refs to functions are treated
|
|
|
|
// like lvalue refs.
|
2009-12-10 07:02:17 +08:00
|
|
|
OverloadingResult ConvOvlResult = OR_Success;
|
2010-07-01 02:13:39 +08:00
|
|
|
bool T1Function = T1->isFunctionType();
|
|
|
|
if (isLValueRef || T1Function) {
|
2011-01-27 15:10:08 +08:00
|
|
|
if (InitCategory.isLValue() &&
|
2011-01-27 08:58:17 +08:00
|
|
|
(RefRelationship >= Sema::Ref_Compatible_With_Added_Qualification ||
|
2011-01-27 15:10:08 +08:00
|
|
|
(Kind.isCStyleOrFunctionalCast() &&
|
2011-01-27 08:58:17 +08:00
|
|
|
RefRelationship == Sema::Ref_Related))) {
|
2011-01-27 15:10:08 +08:00
|
|
|
// - is an lvalue (but is not a bit-field), and "cv1 T1" is
|
2009-12-10 07:02:17 +08:00
|
|
|
// reference-compatible with "cv2 T2," or
|
|
|
|
//
|
2011-01-27 15:10:08 +08:00
|
|
|
// Per C++ [over.best.ics]p2, we don't diagnose whether the lvalue is a
|
2009-12-10 07:02:17 +08:00
|
|
|
// bit-field when we're determining whether the reference initialization
|
2010-01-30 03:14:02 +08:00
|
|
|
// can occur. However, we do pay attention to whether it is a bit-field
|
|
|
|
// to decide whether we're actually binding to a temporary created from
|
|
|
|
// the bit-field.
|
2009-12-10 07:02:17 +08:00
|
|
|
if (DerivedToBase)
|
|
|
|
Sequence.AddDerivedToBaseCastStep(
|
2011-01-27 15:10:08 +08:00
|
|
|
S.Context.getQualifiedType(T1, T2Quals),
|
2010-08-25 18:28:54 +08:00
|
|
|
VK_LValue);
|
2010-08-07 19:51:51 +08:00
|
|
|
else if (ObjCConversion)
|
|
|
|
Sequence.AddObjCObjectConversionStep(
|
|
|
|
S.Context.getQualifiedType(T1, T2Quals));
|
|
|
|
|
2013-04-11 08:58:58 +08:00
|
|
|
ExprValueKind ValueKind =
|
|
|
|
convertQualifiersAndValueKindIfNecessary(S, Sequence, Initializer,
|
|
|
|
cv1T1, T1Quals, T2Quals,
|
|
|
|
isLValueRef);
|
|
|
|
Sequence.AddReferenceBindingStep(cv1T1, ValueKind == VK_RValue);
|
2009-12-10 07:02:17 +08:00
|
|
|
return;
|
|
|
|
}
|
2011-01-27 15:10:08 +08:00
|
|
|
|
|
|
|
// - has a class type (i.e., T2 is a class type), where T1 is not
|
|
|
|
// reference-related to T2, and can be implicitly converted to an
|
|
|
|
// lvalue of type "cv3 T3," where "cv1 T1" is reference-compatible
|
|
|
|
// with "cv3 T3" (this conversion is selected by enumerating the
|
2009-12-10 07:02:17 +08:00
|
|
|
// applicable conversion functions (13.3.1.6) and choosing the best
|
|
|
|
// one through overload resolution (13.3)),
|
2010-07-01 02:13:39 +08:00
|
|
|
// If we have an rvalue ref to function type here, the rhs must be
|
2013-09-22 05:23:47 +08:00
|
|
|
// an rvalue. DR1287 removed the "implicitly" here.
|
2010-07-01 02:13:39 +08:00
|
|
|
if (RefRelationship == Sema::Ref_Incompatible && T2->isRecordType() &&
|
|
|
|
(isLValueRef || InitCategory.isRValue())) {
|
2013-09-22 05:23:47 +08:00
|
|
|
ConvOvlResult = TryRefInitWithConversionFunction(
|
|
|
|
S, Entity, Kind, Initializer, /*AllowRValues*/isRValueRef, Sequence);
|
2009-12-10 07:02:17 +08:00
|
|
|
if (ConvOvlResult == OR_Success)
|
|
|
|
return;
|
2013-09-22 05:23:47 +08:00
|
|
|
if (ConvOvlResult != OR_No_Viable_Function)
|
2010-01-12 08:44:57 +08:00
|
|
|
Sequence.SetOverloadFailure(
|
2013-09-22 05:23:47 +08:00
|
|
|
InitializationSequence::FK_ReferenceInitOverloadFailed,
|
|
|
|
ConvOvlResult);
|
2009-12-10 07:02:17 +08:00
|
|
|
}
|
|
|
|
}
|
2010-07-01 02:13:39 +08:00
|
|
|
|
2011-01-27 15:10:08 +08:00
|
|
|
// - Otherwise, the reference shall be an lvalue reference to a
|
2009-12-10 07:02:17 +08:00
|
|
|
// non-volatile const type (i.e., cv1 shall be const), or the reference
|
2011-01-21 00:08:06 +08:00
|
|
|
// shall be an rvalue reference.
|
2011-01-21 08:52:42 +08:00
|
|
|
if (isLValueRef && !(T1Quals.hasConst() && !T1Quals.hasVolatile())) {
|
2010-11-08 23:20:28 +08:00
|
|
|
if (S.Context.getCanonicalType(T2) == S.Context.OverloadTy)
|
|
|
|
Sequence.SetFailed(InitializationSequence::FK_AddressOfOverloadFailed);
|
|
|
|
else if (ConvOvlResult && !Sequence.getFailedCandidateSet().empty())
|
2009-12-10 07:02:17 +08:00
|
|
|
Sequence.SetOverloadFailure(
|
|
|
|
InitializationSequence::FK_ReferenceInitOverloadFailed,
|
|
|
|
ConvOvlResult);
|
2011-01-21 08:52:42 +08:00
|
|
|
else
|
2010-07-01 02:13:39 +08:00
|
|
|
Sequence.SetFailed(InitCategory.isLValue()
|
2009-12-10 07:02:17 +08:00
|
|
|
? (RefRelationship == Sema::Ref_Related
|
|
|
|
? InitializationSequence::FK_ReferenceInitDropsQualifiers
|
|
|
|
: InitializationSequence::FK_NonConstLValueReferenceBindingToUnrelated)
|
|
|
|
: InitializationSequence::FK_NonConstLValueReferenceBindingToTemporary);
|
2010-07-01 02:13:39 +08:00
|
|
|
|
2009-12-10 07:02:17 +08:00
|
|
|
return;
|
|
|
|
}
|
2010-07-01 02:13:39 +08:00
|
|
|
|
2011-01-21 00:44:54 +08:00
|
|
|
// - If the initializer expression
|
|
|
|
// - is an xvalue, class prvalue, array prvalue, or function lvalue and
|
|
|
|
// "cv1 T1" is reference-compatible with "cv2 T2"
|
|
|
|
// Note: functions are handled below.
|
|
|
|
if (!T1Function &&
|
2011-01-27 08:58:17 +08:00
|
|
|
(RefRelationship >= Sema::Ref_Compatible_With_Added_Qualification ||
|
2011-01-27 15:10:08 +08:00
|
|
|
(Kind.isCStyleOrFunctionalCast() &&
|
2011-01-27 08:58:17 +08:00
|
|
|
RefRelationship == Sema::Ref_Related)) &&
|
2011-01-21 00:44:54 +08:00
|
|
|
(InitCategory.isXValue() ||
|
|
|
|
(InitCategory.isPRValue() && T2->isRecordType()) ||
|
|
|
|
(InitCategory.isPRValue() && T2->isArrayType()))) {
|
|
|
|
ExprValueKind ValueKind = InitCategory.isXValue()? VK_XValue : VK_RValue;
|
|
|
|
if (InitCategory.isPRValue() && T2->isRecordType()) {
|
2010-04-18 15:40:54 +08:00
|
|
|
// The corresponding bullet in C++03 [dcl.init.ref]p5 gives the
|
|
|
|
// compiler the freedom to perform a copy here or bind to the
|
|
|
|
// object, while C++0x requires that we bind directly to the
|
|
|
|
// object. Hence, we always bind to the object without making an
|
|
|
|
// extra copy. However, in C++03 requires that we check for the
|
|
|
|
// presence of a suitable copy constructor:
|
|
|
|
//
|
|
|
|
// The constructor that would be used to make the copy shall
|
|
|
|
// be callable whether or not the copy is actually done.
|
2013-01-02 19:42:31 +08:00
|
|
|
if (!S.getLangOpts().CPlusPlus11 && !S.getLangOpts().MicrosoftExt)
|
2010-04-18 15:40:54 +08:00
|
|
|
Sequence.AddExtraneousCopyToTemporary(cv2T2);
|
2013-01-02 19:42:31 +08:00
|
|
|
else if (S.getLangOpts().CPlusPlus11)
|
2011-10-20 00:55:56 +08:00
|
|
|
CheckCXX98CompatAccessibleCopy(S, Entity, Initializer);
|
2009-12-10 07:02:17 +08:00
|
|
|
}
|
2011-01-27 15:10:08 +08:00
|
|
|
|
2011-01-21 00:44:54 +08:00
|
|
|
if (DerivedToBase)
|
|
|
|
Sequence.AddDerivedToBaseCastStep(S.Context.getQualifiedType(T1, T2Quals),
|
|
|
|
ValueKind);
|
|
|
|
else if (ObjCConversion)
|
|
|
|
Sequence.AddObjCObjectConversionStep(
|
|
|
|
S.Context.getQualifiedType(T1, T2Quals));
|
2011-01-27 15:10:08 +08:00
|
|
|
|
2013-04-11 08:58:58 +08:00
|
|
|
ValueKind = convertQualifiersAndValueKindIfNecessary(S, Sequence,
|
|
|
|
Initializer, cv1T1,
|
|
|
|
T1Quals, T2Quals,
|
|
|
|
isLValueRef);
|
|
|
|
|
|
|
|
Sequence.AddReferenceBindingStep(cv1T1, ValueKind == VK_RValue);
|
2011-01-27 15:10:08 +08:00
|
|
|
return;
|
2011-01-21 00:44:54 +08:00
|
|
|
}
|
2011-01-27 15:10:08 +08:00
|
|
|
|
|
|
|
// - has a class type (i.e., T2 is a class type), where T1 is not
|
|
|
|
// reference-related to T2, and can be implicitly converted to an
|
2011-01-21 00:44:54 +08:00
|
|
|
// xvalue, class prvalue, or function lvalue of type "cv3 T3",
|
|
|
|
// where "cv1 T1" is reference-compatible with "cv3 T3",
|
2013-09-22 05:23:47 +08:00
|
|
|
//
|
|
|
|
// DR1287 removes the "implicitly" here.
|
2011-01-21 00:44:54 +08:00
|
|
|
if (T2->isRecordType()) {
|
2009-12-10 07:02:17 +08:00
|
|
|
if (RefRelationship == Sema::Ref_Incompatible) {
|
2013-09-22 05:23:47 +08:00
|
|
|
ConvOvlResult = TryRefInitWithConversionFunction(
|
|
|
|
S, Entity, Kind, Initializer, /*AllowRValues*/true, Sequence);
|
2009-12-10 07:02:17 +08:00
|
|
|
if (ConvOvlResult)
|
|
|
|
Sequence.SetOverloadFailure(
|
2013-09-22 05:23:47 +08:00
|
|
|
InitializationSequence::FK_ReferenceInitOverloadFailed,
|
|
|
|
ConvOvlResult);
|
2011-01-27 15:10:08 +08:00
|
|
|
|
2009-12-10 07:02:17 +08:00
|
|
|
return;
|
|
|
|
}
|
2011-01-27 15:10:08 +08:00
|
|
|
|
2013-03-27 07:59:23 +08:00
|
|
|
if ((RefRelationship == Sema::Ref_Compatible ||
|
|
|
|
RefRelationship == Sema::Ref_Compatible_With_Added_Qualification) &&
|
|
|
|
isRValueRef && InitCategory.isLValue()) {
|
|
|
|
Sequence.SetFailed(
|
|
|
|
InitializationSequence::FK_RValueReferenceBindingToLValue);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2009-12-10 07:02:17 +08:00
|
|
|
Sequence.SetFailed(InitializationSequence::FK_ReferenceInitDropsQualifiers);
|
|
|
|
return;
|
|
|
|
}
|
2011-01-27 15:09:49 +08:00
|
|
|
|
|
|
|
// - Otherwise, a temporary of type "cv1 T1" is created and initialized
|
2009-12-10 07:02:17 +08:00
|
|
|
// from the initializer expression using the rules for a non-reference
|
2013-06-13 08:57:57 +08:00
|
|
|
// copy-initialization (8.5). The reference is then bound to the
|
2009-12-10 07:02:17 +08:00
|
|
|
// temporary. [...]
|
2010-06-04 10:29:22 +08:00
|
|
|
|
|
|
|
InitializedEntity TempEntity = InitializedEntity::InitializeTemporary(cv1T1);
|
|
|
|
|
2013-06-13 08:57:57 +08:00
|
|
|
// FIXME: Why do we use an implicit conversion here rather than trying
|
|
|
|
// copy-initialization?
|
2011-06-16 07:02:42 +08:00
|
|
|
ImplicitConversionSequence ICS
|
|
|
|
= S.TryImplicitConversion(Initializer, TempEntity.getType(),
|
2013-06-13 08:57:57 +08:00
|
|
|
/*SuppressUserConversions=*/false,
|
|
|
|
/*AllowExplicit=*/false,
|
2011-01-27 08:58:17 +08:00
|
|
|
/*FIXME:InOverloadResolution=*/false,
|
2011-06-16 07:02:42 +08:00
|
|
|
/*CStyle=*/Kind.isCStyleOrFunctionalCast(),
|
|
|
|
/*AllowObjCWritebackConversion=*/false);
|
|
|
|
|
|
|
|
if (ICS.isBad()) {
|
2009-12-10 07:02:17 +08:00
|
|
|
// FIXME: Use the conversion function set stored in ICS to turn
|
|
|
|
// this into an overloading ambiguity diagnostic. However, we need
|
|
|
|
// to keep that set as an OverloadCandidateSet rather than as some
|
|
|
|
// other kind of set.
|
Switch the initialization required by return statements over to the
new InitializationSequence. This fixes some bugs (e.g., PR5808),
changed some diagnostics, and caused more churn than expected. What's
new:
- InitializationSequence now has a "C conversion sequence" category
and step kind, which falls back to
- Changed the diagnostics for returns to always have the result type
of the function first and the type of the expression second.
CheckSingleAssignmentConstraints to peform checking in C.
- Improved ASTs for initialization of return values. The ASTs now
capture all of the temporaries we need to create, but
intentionally do not bind the tempoary that is actually returned,
so that it won't get destroyed twice.
- Make sure to perform an (elidable!) copy of the class object that
is returned from a class.
- Fix copy elision in CodeGen to properly see through the
subexpressions that occur with elidable copies.
- Give "new" its own entity kind; as with return values and thrown
objects, we don't bind the expression so we don't call a
destructor for it.
Note that, with this patch, I've broken returning move-only types in
C++0x. We'll fix it later, when we tackle NRVO.
llvm-svn: 91669
2009-12-18 13:02:21 +08:00
|
|
|
if (ConvOvlResult && !Sequence.getFailedCandidateSet().empty())
|
|
|
|
Sequence.SetOverloadFailure(
|
|
|
|
InitializationSequence::FK_ReferenceInitOverloadFailed,
|
|
|
|
ConvOvlResult);
|
2010-11-08 23:20:28 +08:00
|
|
|
else if (S.Context.getCanonicalType(T2) == S.Context.OverloadTy)
|
|
|
|
Sequence.SetFailed(InitializationSequence::FK_AddressOfOverloadFailed);
|
Switch the initialization required by return statements over to the
new InitializationSequence. This fixes some bugs (e.g., PR5808),
changed some diagnostics, and caused more churn than expected. What's
new:
- InitializationSequence now has a "C conversion sequence" category
and step kind, which falls back to
- Changed the diagnostics for returns to always have the result type
of the function first and the type of the expression second.
CheckSingleAssignmentConstraints to peform checking in C.
- Improved ASTs for initialization of return values. The ASTs now
capture all of the temporaries we need to create, but
intentionally do not bind the tempoary that is actually returned,
so that it won't get destroyed twice.
- Make sure to perform an (elidable!) copy of the class object that
is returned from a class.
- Fix copy elision in CodeGen to properly see through the
subexpressions that occur with elidable copies.
- Give "new" its own entity kind; as with return values and thrown
objects, we don't bind the expression so we don't call a
destructor for it.
Note that, with this patch, I've broken returning move-only types in
C++0x. We'll fix it later, when we tackle NRVO.
llvm-svn: 91669
2009-12-18 13:02:21 +08:00
|
|
|
else
|
|
|
|
Sequence.SetFailed(InitializationSequence::FK_ReferenceInitFailed);
|
2009-12-10 07:02:17 +08:00
|
|
|
return;
|
2011-06-16 07:02:42 +08:00
|
|
|
} else {
|
|
|
|
Sequence.AddConversionSequenceStep(ICS, TempEntity.getType());
|
2009-12-10 07:02:17 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
// [...] 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.
|
2010-01-13 04:32:25 +08:00
|
|
|
unsigned T1CVRQuals = T1Quals.getCVRQualifiers();
|
|
|
|
unsigned T2CVRQuals = T2Quals.getCVRQualifiers();
|
2011-01-27 15:10:08 +08:00
|
|
|
if (RefRelationship == Sema::Ref_Related &&
|
2010-01-13 04:32:25 +08:00
|
|
|
(T1CVRQuals | T2CVRQuals) != T1CVRQuals) {
|
2009-12-10 07:02:17 +08:00
|
|
|
Sequence.SetFailed(InitializationSequence::FK_ReferenceInitDropsQualifiers);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2011-01-27 15:10:08 +08:00
|
|
|
// [...] If T1 is reference-related to T2 and the reference is an rvalue
|
2011-01-21 08:52:42 +08:00
|
|
|
// reference, the initializer expression shall not be an lvalue.
|
2011-01-27 15:10:08 +08:00
|
|
|
if (RefRelationship >= Sema::Ref_Related && !isLValueRef &&
|
2011-01-21 08:52:42 +08:00
|
|
|
InitCategory.isLValue()) {
|
|
|
|
Sequence.SetFailed(
|
|
|
|
InitializationSequence::FK_RValueReferenceBindingToLValue);
|
|
|
|
return;
|
|
|
|
}
|
2011-01-27 15:10:08 +08:00
|
|
|
|
2009-12-10 07:02:17 +08:00
|
|
|
Sequence.AddReferenceBindingStep(cv1T1, /*bindingTemporary=*/true);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
/// \brief Attempt character array initialization from a string literal
|
2011-01-27 15:10:08 +08:00
|
|
|
/// (C++ [dcl.init.string], C99 6.7.8).
|
|
|
|
static void TryStringLiteralInitialization(Sema &S,
|
2009-12-10 07:02:17 +08:00
|
|
|
const InitializedEntity &Entity,
|
|
|
|
const InitializationKind &Kind,
|
|
|
|
Expr *Initializer,
|
|
|
|
InitializationSequence &Sequence) {
|
2009-12-22 23:35:07 +08:00
|
|
|
Sequence.AddStringInitStep(Entity.getType());
|
2009-12-10 07:02:17 +08:00
|
|
|
}
|
|
|
|
|
2009-12-15 08:01:57 +08:00
|
|
|
/// \brief Attempt value initialization (C++ [dcl.init]p7).
|
2011-01-27 15:10:08 +08:00
|
|
|
static void TryValueInitialization(Sema &S,
|
2009-12-15 08:01:57 +08:00
|
|
|
const InitializedEntity &Entity,
|
|
|
|
const InitializationKind &Kind,
|
2012-07-05 16:39:21 +08:00
|
|
|
InitializationSequence &Sequence,
|
|
|
|
InitListExpr *InitList) {
|
|
|
|
assert((!InitList || InitList->getNumInits() == 0) &&
|
|
|
|
"Shouldn't use value-init for non-empty init lists");
|
|
|
|
|
2012-02-15 05:14:13 +08:00
|
|
|
// C++98 [dcl.init]p5, C++11 [dcl.init]p7:
|
2009-12-15 08:01:57 +08:00
|
|
|
//
|
|
|
|
// To value-initialize an object of type T means:
|
2009-12-22 23:35:07 +08:00
|
|
|
QualType T = Entity.getType();
|
2011-01-27 15:10:08 +08:00
|
|
|
|
2009-12-15 08:01:57 +08:00
|
|
|
// -- if T is an array type, then each element is value-initialized;
|
2012-02-15 05:14:13 +08:00
|
|
|
T = S.Context.getBaseElementType(T);
|
2011-01-27 15:10:08 +08:00
|
|
|
|
2009-12-15 08:01:57 +08:00
|
|
|
if (const RecordType *RT = T->getAs<RecordType>()) {
|
|
|
|
if (CXXRecordDecl *ClassDecl = dyn_cast<CXXRecordDecl>(RT->getDecl())) {
|
2012-07-05 16:39:21 +08:00
|
|
|
bool NeedZeroInitialization = true;
|
2013-01-02 19:42:31 +08:00
|
|
|
if (!S.getLangOpts().CPlusPlus11) {
|
2012-07-05 16:39:21 +08:00
|
|
|
// C++98:
|
|
|
|
// -- if T is a class type (clause 9) with a user-declared constructor
|
|
|
|
// (12.1), then the default constructor for T is called (and the
|
|
|
|
// initialization is ill-formed if T has no accessible default
|
|
|
|
// constructor);
|
2012-02-15 05:14:13 +08:00
|
|
|
if (ClassDecl->hasUserDeclaredConstructor())
|
2012-07-05 16:39:21 +08:00
|
|
|
NeedZeroInitialization = false;
|
2012-02-15 05:14:13 +08:00
|
|
|
} else {
|
|
|
|
// C++11:
|
|
|
|
// -- if T is a class type (clause 9) with either no default constructor
|
|
|
|
// (12.1 [class.ctor]) or a default constructor that is user-provided
|
|
|
|
// or deleted, then the object is default-initialized;
|
|
|
|
CXXConstructorDecl *CD = S.LookupDefaultConstructor(ClassDecl);
|
|
|
|
if (!CD || !CD->getCanonicalDecl()->isDefaulted() || CD->isDeleted())
|
2012-07-05 16:39:21 +08:00
|
|
|
NeedZeroInitialization = false;
|
2012-02-15 05:14:13 +08:00
|
|
|
}
|
2011-01-27 15:10:08 +08:00
|
|
|
|
2012-02-15 05:14:13 +08:00
|
|
|
// -- if T is a (possibly cv-qualified) non-union class type without a
|
|
|
|
// user-provided or deleted default constructor, then the object is
|
|
|
|
// zero-initialized and, if T has a non-trivial default constructor,
|
|
|
|
// default-initialized;
|
2012-10-18 08:44:17 +08:00
|
|
|
// The 'non-union' here was removed by DR1502. The 'non-trivial default
|
|
|
|
// constructor' part was removed by DR1507.
|
2012-07-05 16:39:21 +08:00
|
|
|
if (NeedZeroInitialization)
|
|
|
|
Sequence.AddZeroInitializationStep(Entity.getType());
|
|
|
|
|
2012-12-08 10:01:17 +08:00
|
|
|
// C++03:
|
|
|
|
// -- if T is a non-union class type without a user-declared constructor,
|
|
|
|
// then every non-static data member and base class component of T is
|
|
|
|
// value-initialized;
|
|
|
|
// [...] A program that calls for [...] value-initialization of an
|
|
|
|
// entity of reference type is ill-formed.
|
|
|
|
//
|
|
|
|
// C++11 doesn't need this handling, because value-initialization does not
|
|
|
|
// occur recursively there, and the implicit default constructor is
|
|
|
|
// defined as deleted in the problematic cases.
|
2013-01-02 19:42:31 +08:00
|
|
|
if (!S.getLangOpts().CPlusPlus11 &&
|
2012-12-08 10:01:17 +08:00
|
|
|
ClassDecl->hasUninitializedReferenceMember()) {
|
|
|
|
Sequence.SetFailed(InitializationSequence::FK_TooManyInitsForReference);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2012-07-05 16:39:21 +08:00
|
|
|
// If this is list-value-initialization, pass the empty init list on when
|
|
|
|
// building the constructor call. This affects the semantics of a few
|
|
|
|
// things (such as whether an explicit default constructor can be called).
|
|
|
|
Expr *InitListAsExpr = InitList;
|
2013-05-03 23:05:50 +08:00
|
|
|
MultiExprArg Args(&InitListAsExpr, InitList ? 1 : 0);
|
2012-07-05 16:39:21 +08:00
|
|
|
bool InitListSyntax = InitList;
|
|
|
|
|
2013-05-03 23:05:50 +08:00
|
|
|
return TryConstructorInitialization(S, Entity, Kind, Args, T, Sequence,
|
|
|
|
InitListSyntax);
|
2009-12-15 08:01:57 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2009-12-22 23:35:07 +08:00
|
|
|
Sequence.AddZeroInitializationStep(Entity.getType());
|
2009-12-15 08:01:57 +08:00
|
|
|
}
|
|
|
|
|
2009-12-16 09:38:02 +08:00
|
|
|
/// \brief Attempt default initialization (C++ [dcl.init]p6).
|
|
|
|
static void TryDefaultInitialization(Sema &S,
|
|
|
|
const InitializedEntity &Entity,
|
|
|
|
const InitializationKind &Kind,
|
|
|
|
InitializationSequence &Sequence) {
|
|
|
|
assert(Kind.getKind() == InitializationKind::IK_Default);
|
2011-01-27 15:10:08 +08:00
|
|
|
|
2009-12-16 09:38:02 +08:00
|
|
|
// C++ [dcl.init]p6:
|
|
|
|
// To default-initialize an object of type T means:
|
|
|
|
// - if T is an array type, each element is default-initialized;
|
2011-06-16 07:02:42 +08:00
|
|
|
QualType DestType = S.Context.getBaseElementType(Entity.getType());
|
|
|
|
|
2009-12-16 09:38:02 +08:00
|
|
|
// - if T is a (possibly cv-qualified) class type (Clause 9), the default
|
|
|
|
// constructor for T is called (and the initialization is ill-formed if
|
|
|
|
// T has no accessible default constructor);
|
2012-03-11 15:00:24 +08:00
|
|
|
if (DestType->isRecordType() && S.getLangOpts().CPlusPlus) {
|
2013-05-06 04:40:26 +08:00
|
|
|
TryConstructorInitialization(S, Entity, Kind, None, DestType, Sequence);
|
2010-08-23 15:55:51 +08:00
|
|
|
return;
|
2009-12-16 09:38:02 +08:00
|
|
|
}
|
2011-01-27 15:10:08 +08:00
|
|
|
|
2009-12-16 09:38:02 +08:00
|
|
|
// - otherwise, no initialization is performed.
|
2011-01-27 15:10:08 +08:00
|
|
|
|
2009-12-16 09:38:02 +08:00
|
|
|
// If a program calls for the default initialization of an object of
|
2011-01-27 15:10:08 +08:00
|
|
|
// a const-qualified type T, T shall be a class type with a user-provided
|
2009-12-16 09:38:02 +08:00
|
|
|
// default constructor.
|
2012-03-11 15:00:24 +08:00
|
|
|
if (DestType.isConstQualified() && S.getLangOpts().CPlusPlus) {
|
2009-12-16 09:38:02 +08:00
|
|
|
Sequence.SetFailed(InitializationSequence::FK_DefaultInitOfConst);
|
2011-06-16 07:02:42 +08:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
// If the destination type has a lifetime property, zero-initialize it.
|
|
|
|
if (DestType.getQualifiers().hasObjCLifetime()) {
|
|
|
|
Sequence.AddZeroInitializationStep(Entity.getType());
|
|
|
|
return;
|
|
|
|
}
|
2009-12-16 09:38:02 +08:00
|
|
|
}
|
|
|
|
|
2009-12-10 07:02:17 +08:00
|
|
|
/// \brief Attempt a user-defined conversion between two types (C++ [dcl.init]),
|
|
|
|
/// which enumerates all conversion functions and performs overload resolution
|
|
|
|
/// to select the best.
|
2011-01-27 15:10:08 +08:00
|
|
|
static void TryUserDefinedConversion(Sema &S,
|
2009-12-10 07:02:17 +08:00
|
|
|
const InitializedEntity &Entity,
|
|
|
|
const InitializationKind &Kind,
|
|
|
|
Expr *Initializer,
|
2013-09-22 05:19:19 +08:00
|
|
|
InitializationSequence &Sequence,
|
|
|
|
bool TopLevelOfInitList) {
|
2009-12-22 23:35:07 +08:00
|
|
|
QualType DestType = Entity.getType();
|
2009-12-15 01:27:33 +08:00
|
|
|
assert(!DestType->isReferenceType() && "References are handled elsewhere");
|
|
|
|
QualType SourceType = Initializer->getType();
|
|
|
|
assert((DestType->isRecordType() || SourceType->isRecordType()) &&
|
|
|
|
"Must have a class type to perform a user-defined conversion");
|
2011-01-27 15:10:08 +08:00
|
|
|
|
2009-12-15 01:27:33 +08:00
|
|
|
// Build the candidate set directly in the initialization sequence
|
|
|
|
// structure, so that it will persist if we fail.
|
|
|
|
OverloadCandidateSet &CandidateSet = Sequence.getFailedCandidateSet();
|
|
|
|
CandidateSet.clear();
|
2011-01-27 15:10:08 +08:00
|
|
|
|
2009-12-15 01:27:33 +08:00
|
|
|
// Determine whether we are allowed to call explicit constructors or
|
|
|
|
// explicit conversion operators.
|
2012-02-13 00:37:24 +08:00
|
|
|
bool AllowExplicit = Kind.AllowExplicit();
|
2011-01-27 15:10:08 +08:00
|
|
|
|
2009-12-15 01:27:33 +08:00
|
|
|
if (const RecordType *DestRecordType = DestType->getAs<RecordType>()) {
|
|
|
|
// The type we're converting to is a class type. Enumerate its constructors
|
|
|
|
// to see if there is a suitable conversion.
|
|
|
|
CXXRecordDecl *DestRecordDecl
|
|
|
|
= cast<CXXRecordDecl>(DestRecordType->getDecl());
|
2011-01-27 15:10:08 +08:00
|
|
|
|
2010-04-26 22:36:57 +08:00
|
|
|
// Try to complete the type we're converting to.
|
2011-01-27 15:10:08 +08:00
|
|
|
if (!S.RequireCompleteType(Kind.getLocation(), DestType, 0)) {
|
2012-12-19 08:45:41 +08:00
|
|
|
DeclContext::lookup_result R = S.LookupConstructors(DestRecordDecl);
|
2012-10-19 00:57:32 +08:00
|
|
|
// The container holding the constructors can under certain conditions
|
|
|
|
// be changed while iterating. To be safe we copy the lookup results
|
|
|
|
// to a new container.
|
2012-12-19 08:45:41 +08:00
|
|
|
SmallVector<NamedDecl*, 8> CopyOfCon(R.begin(), R.end());
|
2013-07-04 11:08:24 +08:00
|
|
|
for (SmallVectorImpl<NamedDecl *>::iterator
|
2012-10-19 00:57:32 +08:00
|
|
|
Con = CopyOfCon.begin(), ConEnd = CopyOfCon.end();
|
2010-04-26 22:36:57 +08:00
|
|
|
Con != ConEnd; ++Con) {
|
|
|
|
NamedDecl *D = *Con;
|
|
|
|
DeclAccessPair FoundDecl = DeclAccessPair::make(D, D->getAccess());
|
2011-01-27 15:10:08 +08:00
|
|
|
|
2010-04-26 22:36:57 +08:00
|
|
|
// Find the constructor (which may be a template).
|
|
|
|
CXXConstructorDecl *Constructor = 0;
|
|
|
|
FunctionTemplateDecl *ConstructorTmpl
|
|
|
|
= dyn_cast<FunctionTemplateDecl>(D);
|
2009-12-15 01:27:33 +08:00
|
|
|
if (ConstructorTmpl)
|
2010-04-26 22:36:57 +08:00
|
|
|
Constructor = cast<CXXConstructorDecl>(
|
|
|
|
ConstructorTmpl->getTemplatedDecl());
|
2010-07-01 11:43:00 +08:00
|
|
|
else
|
2010-04-26 22:36:57 +08:00
|
|
|
Constructor = cast<CXXConstructorDecl>(D);
|
2011-01-27 15:10:08 +08:00
|
|
|
|
2010-04-26 22:36:57 +08:00
|
|
|
if (!Constructor->isInvalidDecl() &&
|
|
|
|
Constructor->isConvertingConstructor(AllowExplicit)) {
|
|
|
|
if (ConstructorTmpl)
|
|
|
|
S.AddTemplateOverloadCandidate(ConstructorTmpl, FoundDecl,
|
|
|
|
/*ExplicitArgs*/ 0,
|
2012-02-25 19:00:22 +08:00
|
|
|
Initializer, CandidateSet,
|
2010-07-01 11:43:00 +08:00
|
|
|
/*SuppressUserConversions=*/true);
|
2010-04-26 22:36:57 +08:00
|
|
|
else
|
|
|
|
S.AddOverloadCandidate(Constructor, FoundDecl,
|
2012-02-25 19:00:22 +08:00
|
|
|
Initializer, CandidateSet,
|
2010-07-01 11:43:00 +08:00
|
|
|
/*SuppressUserConversions=*/true);
|
2010-04-26 22:36:57 +08:00
|
|
|
}
|
2011-01-27 15:10:08 +08:00
|
|
|
}
|
2010-04-26 22:36:57 +08:00
|
|
|
}
|
2009-12-15 01:27:33 +08:00
|
|
|
}
|
2009-12-19 16:11:05 +08:00
|
|
|
|
|
|
|
SourceLocation DeclLoc = Initializer->getLocStart();
|
|
|
|
|
2009-12-15 01:27:33 +08:00
|
|
|
if (const RecordType *SourceRecordType = SourceType->getAs<RecordType>()) {
|
|
|
|
// The type we're converting from is a class type, enumerate its conversion
|
|
|
|
// functions.
|
2009-12-19 16:11:05 +08:00
|
|
|
|
2009-12-21 06:12:03 +08:00
|
|
|
// We can only enumerate the conversion functions for a complete type; if
|
|
|
|
// the type isn't complete, simply skip this step.
|
|
|
|
if (!S.RequireCompleteType(DeclLoc, SourceType, 0)) {
|
|
|
|
CXXRecordDecl *SourceRecordDecl
|
|
|
|
= cast<CXXRecordDecl>(SourceRecordType->getDecl());
|
2011-01-27 15:10:08 +08:00
|
|
|
|
2012-11-28 11:56:09 +08:00
|
|
|
std::pair<CXXRecordDecl::conversion_iterator,
|
|
|
|
CXXRecordDecl::conversion_iterator>
|
|
|
|
Conversions = SourceRecordDecl->getVisibleConversionFunctions();
|
|
|
|
for (CXXRecordDecl::conversion_iterator
|
|
|
|
I = Conversions.first, E = Conversions.second; I != E; ++I) {
|
2009-12-21 06:12:03 +08:00
|
|
|
NamedDecl *D = *I;
|
|
|
|
CXXRecordDecl *ActingDC = cast<CXXRecordDecl>(D->getDeclContext());
|
|
|
|
if (isa<UsingShadowDecl>(D))
|
|
|
|
D = cast<UsingShadowDecl>(D)->getTargetDecl();
|
2011-01-27 15:10:08 +08:00
|
|
|
|
2009-12-21 06:12:03 +08:00
|
|
|
FunctionTemplateDecl *ConvTemplate = dyn_cast<FunctionTemplateDecl>(D);
|
|
|
|
CXXConversionDecl *Conv;
|
2009-12-15 01:27:33 +08:00
|
|
|
if (ConvTemplate)
|
2009-12-21 06:12:03 +08:00
|
|
|
Conv = cast<CXXConversionDecl>(ConvTemplate->getTemplatedDecl());
|
2009-12-15 01:27:33 +08:00
|
|
|
else
|
2010-03-31 09:36:47 +08:00
|
|
|
Conv = cast<CXXConversionDecl>(D);
|
2011-01-27 15:10:08 +08:00
|
|
|
|
2009-12-21 06:12:03 +08:00
|
|
|
if (AllowExplicit || !Conv->isExplicit()) {
|
|
|
|
if (ConvTemplate)
|
2010-03-19 15:35:19 +08:00
|
|
|
S.AddTemplateConversionCandidate(ConvTemplate, I.getPair(),
|
2010-01-26 09:37:31 +08:00
|
|
|
ActingDC, Initializer, DestType,
|
2013-12-19 05:46:16 +08:00
|
|
|
CandidateSet, AllowExplicit);
|
2009-12-21 06:12:03 +08:00
|
|
|
else
|
2010-03-19 15:35:19 +08:00
|
|
|
S.AddConversionCandidate(Conv, I.getPair(), ActingDC,
|
2013-12-19 05:46:16 +08:00
|
|
|
Initializer, DestType, CandidateSet,
|
|
|
|
AllowExplicit);
|
2009-12-21 06:12:03 +08:00
|
|
|
}
|
2009-12-15 01:27:33 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2011-01-27 15:10:08 +08:00
|
|
|
|
|
|
|
// Perform overload resolution. If it fails, return the failed result.
|
2009-12-15 01:27:33 +08:00
|
|
|
OverloadCandidateSet::iterator Best;
|
2010-01-12 08:44:57 +08:00
|
|
|
if (OverloadingResult Result
|
2010-09-12 16:07:23 +08:00
|
|
|
= CandidateSet.BestViableFunction(S, DeclLoc, Best, true)) {
|
2009-12-15 01:27:33 +08:00
|
|
|
Sequence.SetOverloadFailure(
|
2011-01-27 15:10:08 +08:00
|
|
|
InitializationSequence::FK_UserConversionOverloadFailed,
|
2009-12-15 01:27:33 +08:00
|
|
|
Result);
|
|
|
|
return;
|
|
|
|
}
|
2010-01-12 08:44:57 +08:00
|
|
|
|
2009-12-15 01:27:33 +08:00
|
|
|
FunctionDecl *Function = Best->Function;
|
2013-02-12 16:08:54 +08:00
|
|
|
Function->setReferenced();
|
2011-11-19 19:44:21 +08:00
|
|
|
bool HadMultipleCandidates = (CandidateSet.size() > 1);
|
2011-01-27 15:10:08 +08:00
|
|
|
|
2009-12-15 01:27:33 +08:00
|
|
|
if (isa<CXXConstructorDecl>(Function)) {
|
|
|
|
// Add the user-defined conversion step. Any cv-qualification conversion is
|
2012-02-12 03:22:50 +08:00
|
|
|
// subsumed by the initialization. Per DR5, the created temporary is of the
|
|
|
|
// cv-unqualified type of the destination.
|
|
|
|
Sequence.AddUserConversionStep(Function, Best->FoundDecl,
|
|
|
|
DestType.getUnqualifiedType(),
|
2011-11-19 19:44:21 +08:00
|
|
|
HadMultipleCandidates);
|
2009-12-15 01:27:33 +08:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Add the user-defined conversion step that calls the conversion function.
|
2010-07-13 16:18:22 +08:00
|
|
|
QualType ConvType = Function->getCallResultType();
|
2010-04-18 06:01:05 +08:00
|
|
|
if (ConvType->getAs<RecordType>()) {
|
2012-02-12 03:22:50 +08:00
|
|
|
// If we're converting to a class type, there may be an copy of
|
2010-04-18 06:01:05 +08:00
|
|
|
// the resulting temporary object (possible to create an object of
|
|
|
|
// a base class type). That copy is not a separate conversion, so
|
|
|
|
// we just make a note of the actual destination type (possibly a
|
|
|
|
// base class of the type returned by the conversion function) and
|
|
|
|
// let the user-defined conversion step handle the conversion.
|
2011-11-19 19:44:21 +08:00
|
|
|
Sequence.AddUserConversionStep(Function, Best->FoundDecl, DestType,
|
|
|
|
HadMultipleCandidates);
|
2010-04-18 06:01:05 +08:00
|
|
|
return;
|
|
|
|
}
|
2009-12-15 01:27:33 +08:00
|
|
|
|
2011-11-19 19:44:21 +08:00
|
|
|
Sequence.AddUserConversionStep(Function, Best->FoundDecl, ConvType,
|
|
|
|
HadMultipleCandidates);
|
2011-01-27 15:10:08 +08:00
|
|
|
|
2010-04-18 06:01:05 +08:00
|
|
|
// If the conversion following the call to the conversion function
|
|
|
|
// is interesting, add it as a separate step.
|
2009-12-15 01:27:33 +08:00
|
|
|
if (Best->FinalConversion.First || Best->FinalConversion.Second ||
|
|
|
|
Best->FinalConversion.Third) {
|
|
|
|
ImplicitConversionSequence ICS;
|
2010-01-12 08:44:57 +08:00
|
|
|
ICS.setStandard();
|
2009-12-15 01:27:33 +08:00
|
|
|
ICS.Standard = Best->FinalConversion;
|
2013-09-22 05:19:19 +08:00
|
|
|
Sequence.AddConversionSequenceStep(ICS, DestType, TopLevelOfInitList);
|
2009-12-15 01:27:33 +08:00
|
|
|
}
|
2009-12-10 07:02:17 +08:00
|
|
|
}
|
|
|
|
|
2013-06-20 10:18:31 +08:00
|
|
|
/// An egregious hack for compatibility with libstdc++-4.2: in <tr1/hashtable>,
|
|
|
|
/// a function with a pointer return type contains a 'return false;' statement.
|
|
|
|
/// In C++11, 'false' is not a null pointer, so this breaks the build of any
|
|
|
|
/// code using that header.
|
|
|
|
///
|
|
|
|
/// Work around this by treating 'return false;' as zero-initializing the result
|
|
|
|
/// if it's used in a pointer-returning function in a system header.
|
|
|
|
static bool isLibstdcxxPointerReturnFalseHack(Sema &S,
|
|
|
|
const InitializedEntity &Entity,
|
|
|
|
const Expr *Init) {
|
|
|
|
return S.getLangOpts().CPlusPlus11 &&
|
|
|
|
Entity.getKind() == InitializedEntity::EK_Result &&
|
|
|
|
Entity.getType()->isPointerType() &&
|
|
|
|
isa<CXXBoolLiteralExpr>(Init) &&
|
|
|
|
!cast<CXXBoolLiteralExpr>(Init)->getValue() &&
|
|
|
|
S.getSourceManager().isInSystemHeader(Init->getExprLoc());
|
|
|
|
}
|
|
|
|
|
2011-06-16 07:02:42 +08:00
|
|
|
/// The non-zero enum values here are indexes into diagnostic alternatives.
|
|
|
|
enum InvalidICRKind { IIK_okay, IIK_nonlocal, IIK_nonscalar };
|
|
|
|
|
|
|
|
/// Determines whether this expression is an acceptable ICR source.
|
2011-06-28 07:59:58 +08:00
|
|
|
static InvalidICRKind isInvalidICRSource(ASTContext &C, Expr *e,
|
2012-11-28 07:02:53 +08:00
|
|
|
bool isAddressOf, bool &isWeakAccess) {
|
2011-06-16 07:02:42 +08:00
|
|
|
// Skip parens.
|
|
|
|
e = e->IgnoreParens();
|
|
|
|
|
|
|
|
// Skip address-of nodes.
|
|
|
|
if (UnaryOperator *op = dyn_cast<UnaryOperator>(e)) {
|
|
|
|
if (op->getOpcode() == UO_AddrOf)
|
2012-11-28 07:02:53 +08:00
|
|
|
return isInvalidICRSource(C, op->getSubExpr(), /*addressof*/ true,
|
|
|
|
isWeakAccess);
|
2011-06-16 07:02:42 +08:00
|
|
|
|
|
|
|
// Skip certain casts.
|
2011-06-28 07:59:58 +08:00
|
|
|
} else if (CastExpr *ce = dyn_cast<CastExpr>(e)) {
|
|
|
|
switch (ce->getCastKind()) {
|
2011-06-16 07:02:42 +08:00
|
|
|
case CK_Dependent:
|
|
|
|
case CK_BitCast:
|
|
|
|
case CK_LValueBitCast:
|
|
|
|
case CK_NoOp:
|
2012-11-28 07:02:53 +08:00
|
|
|
return isInvalidICRSource(C, ce->getSubExpr(), isAddressOf, isWeakAccess);
|
2011-06-16 07:02:42 +08:00
|
|
|
|
|
|
|
case CK_ArrayToPointerDecay:
|
|
|
|
return IIK_nonscalar;
|
|
|
|
|
|
|
|
case CK_NullToPointer:
|
|
|
|
return IIK_okay;
|
|
|
|
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
// If we have a declaration reference, it had better be a local variable.
|
2012-03-10 17:33:50 +08:00
|
|
|
} else if (isa<DeclRefExpr>(e)) {
|
2012-11-28 07:02:53 +08:00
|
|
|
// set isWeakAccess to true, to mean that there will be an implicit
|
|
|
|
// load which requires a cleanup.
|
|
|
|
if (e->getType().getObjCLifetime() == Qualifiers::OCL_Weak)
|
|
|
|
isWeakAccess = true;
|
|
|
|
|
2011-06-28 07:59:58 +08:00
|
|
|
if (!isAddressOf) return IIK_nonlocal;
|
|
|
|
|
2012-03-10 17:33:50 +08:00
|
|
|
VarDecl *var = dyn_cast<VarDecl>(cast<DeclRefExpr>(e)->getDecl());
|
|
|
|
if (!var) return IIK_nonlocal;
|
2011-06-28 07:59:58 +08:00
|
|
|
|
|
|
|
return (var->hasLocalStorage() ? IIK_okay : IIK_nonlocal);
|
2011-06-16 07:02:42 +08:00
|
|
|
|
|
|
|
// If we have a conditional operator, check both sides.
|
|
|
|
} else if (ConditionalOperator *cond = dyn_cast<ConditionalOperator>(e)) {
|
2012-11-28 07:02:53 +08:00
|
|
|
if (InvalidICRKind iik = isInvalidICRSource(C, cond->getLHS(), isAddressOf,
|
|
|
|
isWeakAccess))
|
2011-06-16 07:02:42 +08:00
|
|
|
return iik;
|
|
|
|
|
2012-11-28 07:02:53 +08:00
|
|
|
return isInvalidICRSource(C, cond->getRHS(), isAddressOf, isWeakAccess);
|
2011-06-16 07:02:42 +08:00
|
|
|
|
|
|
|
// These are never scalar.
|
|
|
|
} else if (isa<ArraySubscriptExpr>(e)) {
|
|
|
|
return IIK_nonscalar;
|
|
|
|
|
|
|
|
// Otherwise, it needs to be a null pointer constant.
|
|
|
|
} else {
|
|
|
|
return (e->isNullPointerConstant(C, Expr::NPC_ValueDependentIsNull)
|
|
|
|
? IIK_okay : IIK_nonlocal);
|
|
|
|
}
|
|
|
|
|
|
|
|
return IIK_nonlocal;
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Check whether the given expression is a valid operand for an
|
|
|
|
/// indirect copy/restore.
|
|
|
|
static void checkIndirectCopyRestoreSource(Sema &S, Expr *src) {
|
|
|
|
assert(src->isRValue());
|
2012-11-28 07:02:53 +08:00
|
|
|
bool isWeakAccess = false;
|
|
|
|
InvalidICRKind iik = isInvalidICRSource(S.Context, src, false, isWeakAccess);
|
|
|
|
// If isWeakAccess to true, there will be an implicit
|
|
|
|
// load which requires a cleanup.
|
|
|
|
if (S.getLangOpts().ObjCAutoRefCount && isWeakAccess)
|
|
|
|
S.ExprNeedsCleanups = true;
|
|
|
|
|
2011-06-16 07:02:42 +08:00
|
|
|
if (iik == IIK_okay) return;
|
|
|
|
|
|
|
|
S.Diag(src->getExprLoc(), diag::err_arc_nonlocal_writeback)
|
|
|
|
<< ((unsigned) iik - 1) // shift index into diagnostic explanations
|
|
|
|
<< src->getSourceRange();
|
|
|
|
}
|
|
|
|
|
2011-02-23 02:29:51 +08:00
|
|
|
/// \brief Determine whether we have compatible array types for the
|
|
|
|
/// purposes of GNU by-copy array initialization.
|
|
|
|
static bool hasCompatibleArrayTypes(ASTContext &Context,
|
|
|
|
const ArrayType *Dest,
|
|
|
|
const ArrayType *Source) {
|
|
|
|
// If the source and destination array types are equivalent, we're
|
|
|
|
// done.
|
|
|
|
if (Context.hasSameType(QualType(Dest, 0), QualType(Source, 0)))
|
|
|
|
return true;
|
|
|
|
|
|
|
|
// Make sure that the element types are the same.
|
|
|
|
if (!Context.hasSameType(Dest->getElementType(), Source->getElementType()))
|
|
|
|
return false;
|
|
|
|
|
|
|
|
// The only mismatch we allow is when the destination is an
|
|
|
|
// incomplete array type and the source is a constant array type.
|
|
|
|
return Source->isConstantArrayType() && Dest->isIncompleteArrayType();
|
|
|
|
}
|
|
|
|
|
2011-06-16 07:02:42 +08:00
|
|
|
static bool tryObjCWritebackConversion(Sema &S,
|
|
|
|
InitializationSequence &Sequence,
|
|
|
|
const InitializedEntity &Entity,
|
|
|
|
Expr *Initializer) {
|
|
|
|
bool ArrayDecay = false;
|
|
|
|
QualType ArgType = Initializer->getType();
|
|
|
|
QualType ArgPointee;
|
|
|
|
if (const ArrayType *ArgArrayType = S.Context.getAsArrayType(ArgType)) {
|
|
|
|
ArrayDecay = true;
|
|
|
|
ArgPointee = ArgArrayType->getElementType();
|
|
|
|
ArgType = S.Context.getPointerType(ArgPointee);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Handle write-back conversion.
|
|
|
|
QualType ConvertedArgType;
|
|
|
|
if (!S.isObjCWritebackConversion(ArgType, Entity.getType(),
|
|
|
|
ConvertedArgType))
|
|
|
|
return false;
|
|
|
|
|
|
|
|
// We should copy unless we're passing to an argument explicitly
|
|
|
|
// marked 'out'.
|
|
|
|
bool ShouldCopy = true;
|
|
|
|
if (ParmVarDecl *param = cast_or_null<ParmVarDecl>(Entity.getDecl()))
|
|
|
|
ShouldCopy = (param->getObjCDeclQualifier() != ParmVarDecl::OBJC_TQ_Out);
|
|
|
|
|
|
|
|
// Do we need an lvalue conversion?
|
|
|
|
if (ArrayDecay || Initializer->isGLValue()) {
|
|
|
|
ImplicitConversionSequence ICS;
|
|
|
|
ICS.setStandard();
|
|
|
|
ICS.Standard.setAsIdentityConversion();
|
|
|
|
|
|
|
|
QualType ResultType;
|
|
|
|
if (ArrayDecay) {
|
|
|
|
ICS.Standard.First = ICK_Array_To_Pointer;
|
|
|
|
ResultType = S.Context.getPointerType(ArgPointee);
|
|
|
|
} else {
|
|
|
|
ICS.Standard.First = ICK_Lvalue_To_Rvalue;
|
|
|
|
ResultType = Initializer->getType().getNonLValueExprType(S.Context);
|
|
|
|
}
|
|
|
|
|
|
|
|
Sequence.AddConversionSequenceStep(ICS, ResultType);
|
|
|
|
}
|
|
|
|
|
|
|
|
Sequence.AddPassByIndirectCopyRestoreStep(Entity.getType(), ShouldCopy);
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2013-02-07 18:55:47 +08:00
|
|
|
static bool TryOCLSamplerInitialization(Sema &S,
|
|
|
|
InitializationSequence &Sequence,
|
|
|
|
QualType DestType,
|
|
|
|
Expr *Initializer) {
|
|
|
|
if (!S.getLangOpts().OpenCL || !DestType->isSamplerT() ||
|
|
|
|
!Initializer->isIntegerConstantExpr(S.getASTContext()))
|
|
|
|
return false;
|
|
|
|
|
|
|
|
Sequence.AddOCLSamplerInitStep(DestType);
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2013-01-20 20:31:11 +08:00
|
|
|
//
|
|
|
|
// OpenCL 1.2 spec, s6.12.10
|
|
|
|
//
|
|
|
|
// The event argument can also be used to associate the
|
|
|
|
// async_work_group_copy with a previous async copy allowing
|
|
|
|
// an event to be shared by multiple async copies; otherwise
|
|
|
|
// event should be zero.
|
|
|
|
//
|
|
|
|
static bool TryOCLZeroEventInitialization(Sema &S,
|
|
|
|
InitializationSequence &Sequence,
|
|
|
|
QualType DestType,
|
|
|
|
Expr *Initializer) {
|
|
|
|
if (!S.getLangOpts().OpenCL || !DestType->isEventT() ||
|
|
|
|
!Initializer->isIntegerConstantExpr(S.getASTContext()) ||
|
|
|
|
(Initializer->EvaluateKnownConstInt(S.getASTContext()) != 0))
|
|
|
|
return false;
|
|
|
|
|
|
|
|
Sequence.AddOCLZeroEventStep(DestType);
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2009-12-10 07:02:17 +08:00
|
|
|
InitializationSequence::InitializationSequence(Sema &S,
|
|
|
|
const InitializedEntity &Entity,
|
|
|
|
const InitializationKind &Kind,
|
2013-09-22 05:19:19 +08:00
|
|
|
MultiExprArg Args,
|
|
|
|
bool TopLevelOfInitList)
|
2010-02-09 07:07:23 +08:00
|
|
|
: FailedCandidateSet(Kind.getLocation()) {
|
2013-09-22 05:55:46 +08:00
|
|
|
InitializeFrom(S, Entity, Kind, Args, TopLevelOfInitList);
|
|
|
|
}
|
|
|
|
|
|
|
|
void InitializationSequence::InitializeFrom(Sema &S,
|
|
|
|
const InitializedEntity &Entity,
|
|
|
|
const InitializationKind &Kind,
|
|
|
|
MultiExprArg Args,
|
|
|
|
bool TopLevelOfInitList) {
|
2011-07-15 06:58:04 +08:00
|
|
|
ASTContext &Context = S.Context;
|
2011-01-27 15:10:08 +08:00
|
|
|
|
2013-04-16 15:28:30 +08:00
|
|
|
// Eliminate non-overload placeholder types in the arguments. We
|
|
|
|
// need to do this before checking whether types are dependent
|
|
|
|
// because lowering a pseudo-object expression might well give us
|
|
|
|
// something of dependent type.
|
2013-05-03 23:05:50 +08:00
|
|
|
for (unsigned I = 0, E = Args.size(); I != E; ++I)
|
2013-04-16 15:28:30 +08:00
|
|
|
if (Args[I]->getType()->isNonOverloadPlaceholderType()) {
|
|
|
|
// FIXME: should we be doing this here?
|
|
|
|
ExprResult result = S.CheckPlaceholderExpr(Args[I]);
|
|
|
|
if (result.isInvalid()) {
|
|
|
|
SetFailed(FK_PlaceholderType);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
Args[I] = result.take();
|
|
|
|
}
|
|
|
|
|
2009-12-10 07:02:17 +08:00
|
|
|
// C++0x [dcl.init]p16:
|
2011-01-27 15:10:08 +08:00
|
|
|
// The semantics of initializers are as follows. The destination type is
|
|
|
|
// the type of the object or reference being initialized and the source
|
2009-12-10 07:02:17 +08:00
|
|
|
// type is the type of the initializer expression. The source type is not
|
2011-01-27 15:10:08 +08:00
|
|
|
// defined when the initializer is a braced-init-list or when it is a
|
2009-12-10 07:02:17 +08:00
|
|
|
// parenthesized list of expressions.
|
2011-07-15 06:58:04 +08:00
|
|
|
QualType DestType = Entity.getType();
|
2009-12-10 07:02:17 +08:00
|
|
|
|
2011-07-15 06:58:04 +08:00
|
|
|
if (DestType->isDependentType() ||
|
2013-05-03 23:05:50 +08:00
|
|
|
Expr::hasAnyTypeDependentArguments(Args)) {
|
2009-12-10 07:02:17 +08:00
|
|
|
SequenceKind = DependentSequence;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2011-06-05 21:59:11 +08:00
|
|
|
// Almost everything is a normal sequence.
|
|
|
|
setSequenceKind(NormalSequence);
|
|
|
|
|
2009-12-10 07:02:17 +08:00
|
|
|
QualType SourceType;
|
|
|
|
Expr *Initializer = 0;
|
2013-05-03 23:05:50 +08:00
|
|
|
if (Args.size() == 1) {
|
2009-12-10 07:02:17 +08:00
|
|
|
Initializer = Args[0];
|
2013-12-19 05:04:43 +08:00
|
|
|
if (S.getLangOpts().ObjC1) {
|
|
|
|
if (S.CheckObjCBridgeRelatedConversions(Initializer->getLocStart(),
|
|
|
|
DestType, Initializer->getType(),
|
|
|
|
Initializer) ||
|
|
|
|
S.ConversionToObjCStringLiteralCheck(DestType, Initializer))
|
|
|
|
Args[0] = Initializer;
|
|
|
|
|
|
|
|
}
|
2009-12-10 07:02:17 +08:00
|
|
|
if (!isa<InitListExpr>(Initializer))
|
|
|
|
SourceType = Initializer->getType();
|
|
|
|
}
|
2011-01-27 15:10:08 +08:00
|
|
|
|
2012-02-13 00:37:36 +08:00
|
|
|
// - If the initializer is a (non-parenthesized) braced-init-list, the
|
|
|
|
// object is list-initialized (8.5.4).
|
|
|
|
if (Kind.getKind() != InitializationKind::IK_Direct) {
|
|
|
|
if (InitListExpr *InitList = dyn_cast_or_null<InitListExpr>(Initializer)) {
|
|
|
|
TryListInitialization(S, Entity, Kind, InitList, *this);
|
|
|
|
return;
|
|
|
|
}
|
2009-12-10 07:02:17 +08:00
|
|
|
}
|
2011-01-27 15:10:08 +08:00
|
|
|
|
2009-12-10 07:02:17 +08:00
|
|
|
// - If the destination type is a reference type, see 8.5.3.
|
|
|
|
if (DestType->isReferenceType()) {
|
|
|
|
// C++0x [dcl.init.ref]p1:
|
|
|
|
// A variable declared to be a T& or T&&, that is, "reference to type T"
|
|
|
|
// (8.3.2), shall be initialized by an object, or function, of type T or
|
|
|
|
// by an object that can be converted into a T.
|
|
|
|
// (Therefore, multiple arguments are not permitted.)
|
2013-05-03 23:05:50 +08:00
|
|
|
if (Args.size() != 1)
|
2011-07-15 06:58:04 +08:00
|
|
|
SetFailed(FK_TooManyInitsForReference);
|
2009-12-10 07:02:17 +08:00
|
|
|
else
|
2011-07-15 06:58:04 +08:00
|
|
|
TryReferenceInitialization(S, Entity, Kind, Args[0], *this);
|
2009-12-10 07:02:17 +08:00
|
|
|
return;
|
|
|
|
}
|
2011-01-27 15:10:08 +08:00
|
|
|
|
2009-12-10 07:02:17 +08:00
|
|
|
// - If the initializer is (), the object is value-initialized.
|
2009-12-16 09:38:02 +08:00
|
|
|
if (Kind.getKind() == InitializationKind::IK_Value ||
|
2013-05-03 23:05:50 +08:00
|
|
|
(Kind.getKind() == InitializationKind::IK_Direct && Args.empty())) {
|
2011-07-15 06:58:04 +08:00
|
|
|
TryValueInitialization(S, Entity, Kind, *this);
|
2009-12-10 07:02:17 +08:00
|
|
|
return;
|
|
|
|
}
|
2011-01-27 15:10:08 +08:00
|
|
|
|
2009-12-16 09:38:02 +08:00
|
|
|
// Handle default initialization.
|
2010-11-20 09:29:55 +08:00
|
|
|
if (Kind.getKind() == InitializationKind::IK_Default) {
|
2011-07-15 06:58:04 +08:00
|
|
|
TryDefaultInitialization(S, Entity, Kind, *this);
|
2009-12-16 09:38:02 +08:00
|
|
|
return;
|
|
|
|
}
|
Switch the initialization required by return statements over to the
new InitializationSequence. This fixes some bugs (e.g., PR5808),
changed some diagnostics, and caused more churn than expected. What's
new:
- InitializationSequence now has a "C conversion sequence" category
and step kind, which falls back to
- Changed the diagnostics for returns to always have the result type
of the function first and the type of the expression second.
CheckSingleAssignmentConstraints to peform checking in C.
- Improved ASTs for initialization of return values. The ASTs now
capture all of the temporaries we need to create, but
intentionally do not bind the tempoary that is actually returned,
so that it won't get destroyed twice.
- Make sure to perform an (elidable!) copy of the class object that
is returned from a class.
- Fix copy elision in CodeGen to properly see through the
subexpressions that occur with elidable copies.
- Give "new" its own entity kind; as with return values and thrown
objects, we don't bind the expression so we don't call a
destructor for it.
Note that, with this patch, I've broken returning move-only types in
C++0x. We'll fix it later, when we tackle NRVO.
llvm-svn: 91669
2009-12-18 13:02:21 +08:00
|
|
|
|
2011-02-21 15:22:22 +08:00
|
|
|
// - If the destination type is an array of characters, an array of
|
|
|
|
// char16_t, an array of char32_t, or an array of wchar_t, and the
|
|
|
|
// initializer is a string literal, see 8.5.2.
|
2011-01-27 15:10:08 +08:00
|
|
|
// - Otherwise, if the destination type is an array, the program is
|
2009-12-10 07:02:17 +08:00
|
|
|
// ill-formed.
|
2011-02-23 02:29:51 +08:00
|
|
|
if (const ArrayType *DestAT = Context.getAsArrayType(DestType)) {
|
2012-01-05 08:13:19 +08:00
|
|
|
if (Initializer && isa<VariableArrayType>(DestAT)) {
|
|
|
|
SetFailed(FK_VariableLengthArrayHasInitializer);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2013-05-15 19:03:04 +08:00
|
|
|
if (Initializer) {
|
|
|
|
switch (IsStringInit(Initializer, DestAT, Context)) {
|
|
|
|
case SIF_None:
|
|
|
|
TryStringLiteralInitialization(S, Entity, Kind, Initializer, *this);
|
|
|
|
return;
|
|
|
|
case SIF_NarrowStringIntoWideChar:
|
|
|
|
SetFailed(FK_NarrowStringIntoWideCharArray);
|
|
|
|
return;
|
|
|
|
case SIF_WideStringIntoChar:
|
|
|
|
SetFailed(FK_WideStringIntoCharArray);
|
|
|
|
return;
|
|
|
|
case SIF_IncompatWideStringIntoWideChar:
|
|
|
|
SetFailed(FK_IncompatWideStringIntoWideChar);
|
|
|
|
return;
|
|
|
|
case SIF_Other:
|
|
|
|
break;
|
|
|
|
}
|
2011-02-21 15:22:22 +08:00
|
|
|
}
|
|
|
|
|
2011-02-23 02:29:51 +08:00
|
|
|
// Note: as an GNU C extension, we allow initialization of an
|
|
|
|
// array from a compound literal that creates an array of the same
|
|
|
|
// type, so long as the initializer has no side effects.
|
2012-03-11 15:00:24 +08:00
|
|
|
if (!S.getLangOpts().CPlusPlus && Initializer &&
|
2011-02-23 02:29:51 +08:00
|
|
|
isa<CompoundLiteralExpr>(Initializer->IgnoreParens()) &&
|
|
|
|
Initializer->getType()->isArrayType()) {
|
|
|
|
const ArrayType *SourceAT
|
|
|
|
= Context.getAsArrayType(Initializer->getType());
|
|
|
|
if (!hasCompatibleArrayTypes(S.Context, DestAT, SourceAT))
|
2011-07-15 06:58:04 +08:00
|
|
|
SetFailed(FK_ArrayTypeMismatch);
|
2011-02-23 02:29:51 +08:00
|
|
|
else if (Initializer->HasSideEffects(S.Context))
|
2011-07-15 06:58:04 +08:00
|
|
|
SetFailed(FK_NonConstantArrayInit);
|
2011-02-23 02:29:51 +08:00
|
|
|
else {
|
2011-07-15 06:58:04 +08:00
|
|
|
AddArrayInitStep(DestType);
|
2011-02-23 02:29:51 +08:00
|
|
|
}
|
2012-02-16 06:38:09 +08:00
|
|
|
}
|
2012-07-05 16:39:21 +08:00
|
|
|
// Note: as a GNU C++ extension, we allow list-initialization of a
|
|
|
|
// class member of array type from a parenthesized initializer list.
|
2012-03-11 15:00:24 +08:00
|
|
|
else if (S.getLangOpts().CPlusPlus &&
|
2012-02-16 06:38:09 +08:00
|
|
|
Entity.getKind() == InitializedEntity::EK_Member &&
|
|
|
|
Initializer && isa<InitListExpr>(Initializer)) {
|
|
|
|
TryListInitialization(S, Entity, Kind, cast<InitListExpr>(Initializer),
|
|
|
|
*this);
|
|
|
|
AddParenthesizedArrayInitStep(DestType);
|
2013-05-15 19:03:04 +08:00
|
|
|
} else if (DestAT->getElementType()->isCharType())
|
2011-07-15 06:58:04 +08:00
|
|
|
SetFailed(FK_ArrayNeedsInitListOrStringLiteral);
|
2013-05-15 19:03:04 +08:00
|
|
|
else if (IsWideCharCompatible(DestAT->getElementType(), Context))
|
|
|
|
SetFailed(FK_ArrayNeedsInitListOrWideStringLiteral);
|
2009-12-10 07:02:17 +08:00
|
|
|
else
|
2011-07-15 06:58:04 +08:00
|
|
|
SetFailed(FK_ArrayNeedsInitList);
|
2011-01-27 15:10:08 +08:00
|
|
|
|
2009-12-10 07:02:17 +08:00
|
|
|
return;
|
|
|
|
}
|
2009-12-19 16:11:05 +08:00
|
|
|
|
2011-06-16 07:02:42 +08:00
|
|
|
// Determine whether we should consider writeback conversions for
|
|
|
|
// Objective-C ARC.
|
2012-03-11 15:00:24 +08:00
|
|
|
bool allowObjCWritebackConversion = S.getLangOpts().ObjCAutoRefCount &&
|
2013-08-01 02:21:45 +08:00
|
|
|
Entity.isParameterKind();
|
2011-06-16 07:02:42 +08:00
|
|
|
|
|
|
|
// We're at the end of the line for C: it's either a write-back conversion
|
|
|
|
// or it's a C assignment. There's no need to check anything else.
|
2012-03-11 15:00:24 +08:00
|
|
|
if (!S.getLangOpts().CPlusPlus) {
|
2011-06-16 07:02:42 +08:00
|
|
|
// If allowed, check whether this is an Objective-C writeback conversion.
|
|
|
|
if (allowObjCWritebackConversion &&
|
2011-07-15 06:58:04 +08:00
|
|
|
tryObjCWritebackConversion(S, *this, Entity, Initializer)) {
|
2011-06-16 07:02:42 +08:00
|
|
|
return;
|
|
|
|
}
|
2013-02-07 18:55:47 +08:00
|
|
|
|
|
|
|
if (TryOCLSamplerInitialization(S, *this, DestType, Initializer))
|
|
|
|
return;
|
2013-01-20 20:31:11 +08:00
|
|
|
|
|
|
|
if (TryOCLZeroEventInitialization(S, *this, DestType, Initializer))
|
|
|
|
return;
|
|
|
|
|
2011-06-16 07:02:42 +08:00
|
|
|
// Handle initialization in C
|
2011-07-15 06:58:04 +08:00
|
|
|
AddCAssignmentStep(DestType);
|
|
|
|
MaybeProduceObjCObject(S, *this, Entity);
|
2009-12-19 16:11:05 +08:00
|
|
|
return;
|
|
|
|
}
|
2011-01-27 15:10:08 +08:00
|
|
|
|
2012-03-11 15:00:24 +08:00
|
|
|
assert(S.getLangOpts().CPlusPlus);
|
2011-06-16 07:02:42 +08:00
|
|
|
|
2009-12-10 07:02:17 +08:00
|
|
|
// - If the destination type is a (possibly cv-qualified) class type:
|
|
|
|
if (DestType->isRecordType()) {
|
2011-01-27 15:10:08 +08:00
|
|
|
// - 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
|
2009-12-10 07:02:17 +08:00
|
|
|
// class of the destination, constructors are considered. [...]
|
|
|
|
if (Kind.getKind() == InitializationKind::IK_Direct ||
|
|
|
|
(Kind.getKind() == InitializationKind::IK_Copy &&
|
|
|
|
(Context.hasSameUnqualifiedType(SourceType, DestType) ||
|
|
|
|
S.IsDerivedFrom(SourceType, DestType))))
|
2013-05-03 23:05:50 +08:00
|
|
|
TryConstructorInitialization(S, Entity, Kind, Args,
|
2011-07-15 06:58:04 +08:00
|
|
|
Entity.getType(), *this);
|
2011-01-27 15:10:08 +08:00
|
|
|
// - Otherwise (i.e., for the remaining copy-initialization cases),
|
2009-12-10 07:02:17 +08:00
|
|
|
// user-defined conversion sequences that can convert from the source
|
2011-01-27 15:10:08 +08:00
|
|
|
// type to the destination type or (when a conversion function is
|
2009-12-10 07:02:17 +08:00
|
|
|
// used) to a derived class thereof are enumerated as described in
|
|
|
|
// 13.3.1.4, and the best one is chosen through overload resolution
|
|
|
|
// (13.3).
|
|
|
|
else
|
2013-09-22 05:19:19 +08:00
|
|
|
TryUserDefinedConversion(S, Entity, Kind, Initializer, *this,
|
|
|
|
TopLevelOfInitList);
|
2009-12-10 07:02:17 +08:00
|
|
|
return;
|
|
|
|
}
|
2011-01-27 15:10:08 +08:00
|
|
|
|
2013-05-03 23:05:50 +08:00
|
|
|
if (Args.size() > 1) {
|
2011-07-15 06:58:04 +08:00
|
|
|
SetFailed(FK_TooManyInitsForScalar);
|
2009-12-16 09:38:02 +08:00
|
|
|
return;
|
|
|
|
}
|
2013-05-03 23:05:50 +08:00
|
|
|
assert(Args.size() == 1 && "Zero-argument case handled above");
|
2011-01-27 15:10:08 +08:00
|
|
|
|
|
|
|
// - Otherwise, if the source type is a (possibly cv-qualified) class
|
2009-12-10 07:02:17 +08:00
|
|
|
// type, conversion functions are considered.
|
2009-12-16 09:38:02 +08:00
|
|
|
if (!SourceType.isNull() && SourceType->isRecordType()) {
|
2013-09-22 05:19:19 +08:00
|
|
|
TryUserDefinedConversion(S, Entity, Kind, Initializer, *this,
|
|
|
|
TopLevelOfInitList);
|
2011-07-15 06:58:04 +08:00
|
|
|
MaybeProduceObjCObject(S, *this, Entity);
|
2009-12-10 07:02:17 +08:00
|
|
|
return;
|
|
|
|
}
|
2011-01-27 15:10:08 +08:00
|
|
|
|
2009-12-10 07:02:17 +08:00
|
|
|
// - Otherwise, the initial value of the object being initialized is the
|
2009-12-15 01:27:33 +08:00
|
|
|
// (possibly converted) value of the initializer expression. Standard
|
2009-12-10 07:02:17 +08:00
|
|
|
// conversions (Clause 4) will be used, if necessary, to convert the
|
2011-01-27 15:10:08 +08:00
|
|
|
// initializer expression to the cv-unqualified version of the
|
2009-12-10 07:02:17 +08:00
|
|
|
// destination type; no user-defined conversions are considered.
|
2011-06-16 07:02:42 +08:00
|
|
|
|
|
|
|
ImplicitConversionSequence ICS
|
|
|
|
= S.TryImplicitConversion(Initializer, Entity.getType(),
|
|
|
|
/*SuppressUserConversions*/true,
|
2010-06-04 10:29:22 +08:00
|
|
|
/*AllowExplicitConversions*/ false,
|
2011-01-27 08:58:17 +08:00
|
|
|
/*InOverloadResolution*/ false,
|
2011-06-16 07:02:42 +08:00
|
|
|
/*CStyle=*/Kind.isCStyleOrFunctionalCast(),
|
|
|
|
allowObjCWritebackConversion);
|
|
|
|
|
|
|
|
if (ICS.isStandard() &&
|
|
|
|
ICS.Standard.Second == ICK_Writeback_Conversion) {
|
|
|
|
// Objective-C ARC writeback conversion.
|
|
|
|
|
|
|
|
// We should copy unless we're passing to an argument explicitly
|
|
|
|
// marked 'out'.
|
|
|
|
bool ShouldCopy = true;
|
|
|
|
if (ParmVarDecl *Param = cast_or_null<ParmVarDecl>(Entity.getDecl()))
|
|
|
|
ShouldCopy = (Param->getObjCDeclQualifier() != ParmVarDecl::OBJC_TQ_Out);
|
|
|
|
|
|
|
|
// If there was an lvalue adjustment, add it as a separate conversion.
|
|
|
|
if (ICS.Standard.First == ICK_Array_To_Pointer ||
|
|
|
|
ICS.Standard.First == ICK_Lvalue_To_Rvalue) {
|
|
|
|
ImplicitConversionSequence LvalueICS;
|
|
|
|
LvalueICS.setStandard();
|
|
|
|
LvalueICS.Standard.setAsIdentityConversion();
|
|
|
|
LvalueICS.Standard.setAllToTypes(ICS.Standard.getToType(0));
|
|
|
|
LvalueICS.Standard.First = ICS.Standard.First;
|
2011-07-15 06:58:04 +08:00
|
|
|
AddConversionSequenceStep(LvalueICS, ICS.Standard.getToType(0));
|
2011-06-16 07:02:42 +08:00
|
|
|
}
|
2011-07-15 06:58:04 +08:00
|
|
|
|
|
|
|
AddPassByIndirectCopyRestoreStep(Entity.getType(), ShouldCopy);
|
2011-06-16 07:02:42 +08:00
|
|
|
} else if (ICS.isBad()) {
|
2011-02-20 05:32:49 +08:00
|
|
|
DeclAccessPair dap;
|
2013-06-20 10:18:31 +08:00
|
|
|
if (isLibstdcxxPointerReturnFalseHack(S, Entity, Initializer)) {
|
|
|
|
AddZeroInitializationStep(Entity.getType());
|
|
|
|
} else if (Initializer->getType() == Context.OverloadTy &&
|
|
|
|
!S.ResolveAddressOfOverloadedFunction(Initializer, DestType,
|
|
|
|
false, dap))
|
2011-07-15 06:58:04 +08:00
|
|
|
SetFailed(InitializationSequence::FK_AddressOfOverloadFailed);
|
2010-11-08 11:40:48 +08:00
|
|
|
else
|
2011-07-15 06:58:04 +08:00
|
|
|
SetFailed(InitializationSequence::FK_ConversionFailed);
|
2011-06-16 07:02:42 +08:00
|
|
|
} else {
|
2013-09-22 05:19:19 +08:00
|
|
|
AddConversionSequenceStep(ICS, Entity.getType(), TopLevelOfInitList);
|
2011-06-17 07:24:51 +08:00
|
|
|
|
2011-07-15 06:58:04 +08:00
|
|
|
MaybeProduceObjCObject(S, *this, Entity);
|
2010-11-08 11:40:48 +08:00
|
|
|
}
|
2009-12-10 07:02:17 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
InitializationSequence::~InitializationSequence() {
|
2011-07-23 18:55:15 +08:00
|
|
|
for (SmallVectorImpl<Step>::iterator Step = Steps.begin(),
|
2009-12-10 07:02:17 +08:00
|
|
|
StepEnd = Steps.end();
|
|
|
|
Step != StepEnd; ++Step)
|
|
|
|
Step->Destroy();
|
|
|
|
}
|
|
|
|
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
// Perform initialization
|
|
|
|
//===----------------------------------------------------------------------===//
|
2011-01-27 15:10:08 +08:00
|
|
|
static Sema::AssignmentAction
|
2013-08-01 07:19:34 +08:00
|
|
|
getAssignmentAction(const InitializedEntity &Entity, bool Diagnose = false) {
|
Switch the initialization required by return statements over to the
new InitializationSequence. This fixes some bugs (e.g., PR5808),
changed some diagnostics, and caused more churn than expected. What's
new:
- InitializationSequence now has a "C conversion sequence" category
and step kind, which falls back to
- Changed the diagnostics for returns to always have the result type
of the function first and the type of the expression second.
CheckSingleAssignmentConstraints to peform checking in C.
- Improved ASTs for initialization of return values. The ASTs now
capture all of the temporaries we need to create, but
intentionally do not bind the tempoary that is actually returned,
so that it won't get destroyed twice.
- Make sure to perform an (elidable!) copy of the class object that
is returned from a class.
- Fix copy elision in CodeGen to properly see through the
subexpressions that occur with elidable copies.
- Give "new" its own entity kind; as with return values and thrown
objects, we don't bind the expression so we don't call a
destructor for it.
Note that, with this patch, I've broken returning move-only types in
C++0x. We'll fix it later, when we tackle NRVO.
llvm-svn: 91669
2009-12-18 13:02:21 +08:00
|
|
|
switch(Entity.getKind()) {
|
|
|
|
case InitializedEntity::EK_Variable:
|
|
|
|
case InitializedEntity::EK_New:
|
2010-12-03 05:47:04 +08:00
|
|
|
case InitializedEntity::EK_Exception:
|
|
|
|
case InitializedEntity::EK_Base:
|
2011-05-01 15:04:31 +08:00
|
|
|
case InitializedEntity::EK_Delegating:
|
Switch the initialization required by return statements over to the
new InitializationSequence. This fixes some bugs (e.g., PR5808),
changed some diagnostics, and caused more churn than expected. What's
new:
- InitializationSequence now has a "C conversion sequence" category
and step kind, which falls back to
- Changed the diagnostics for returns to always have the result type
of the function first and the type of the expression second.
CheckSingleAssignmentConstraints to peform checking in C.
- Improved ASTs for initialization of return values. The ASTs now
capture all of the temporaries we need to create, but
intentionally do not bind the tempoary that is actually returned,
so that it won't get destroyed twice.
- Make sure to perform an (elidable!) copy of the class object that
is returned from a class.
- Fix copy elision in CodeGen to properly see through the
subexpressions that occur with elidable copies.
- Give "new" its own entity kind; as with return values and thrown
objects, we don't bind the expression so we don't call a
destructor for it.
Note that, with this patch, I've broken returning move-only types in
C++0x. We'll fix it later, when we tackle NRVO.
llvm-svn: 91669
2009-12-18 13:02:21 +08:00
|
|
|
return Sema::AA_Initializing;
|
|
|
|
|
|
|
|
case InitializedEntity::EK_Parameter:
|
2011-01-27 15:10:08 +08:00
|
|
|
if (Entity.getDecl() &&
|
2010-04-22 07:24:10 +08:00
|
|
|
isa<ObjCMethodDecl>(Entity.getDecl()->getDeclContext()))
|
|
|
|
return Sema::AA_Sending;
|
|
|
|
|
Switch the initialization required by return statements over to the
new InitializationSequence. This fixes some bugs (e.g., PR5808),
changed some diagnostics, and caused more churn than expected. What's
new:
- InitializationSequence now has a "C conversion sequence" category
and step kind, which falls back to
- Changed the diagnostics for returns to always have the result type
of the function first and the type of the expression second.
CheckSingleAssignmentConstraints to peform checking in C.
- Improved ASTs for initialization of return values. The ASTs now
capture all of the temporaries we need to create, but
intentionally do not bind the tempoary that is actually returned,
so that it won't get destroyed twice.
- Make sure to perform an (elidable!) copy of the class object that
is returned from a class.
- Fix copy elision in CodeGen to properly see through the
subexpressions that occur with elidable copies.
- Give "new" its own entity kind; as with return values and thrown
objects, we don't bind the expression so we don't call a
destructor for it.
Note that, with this patch, I've broken returning move-only types in
C++0x. We'll fix it later, when we tackle NRVO.
llvm-svn: 91669
2009-12-18 13:02:21 +08:00
|
|
|
return Sema::AA_Passing;
|
|
|
|
|
2013-08-01 07:19:34 +08:00
|
|
|
case InitializedEntity::EK_Parameter_CF_Audited:
|
|
|
|
if (Entity.getDecl() &&
|
|
|
|
isa<ObjCMethodDecl>(Entity.getDecl()->getDeclContext()))
|
|
|
|
return Sema::AA_Sending;
|
|
|
|
|
|
|
|
return !Diagnose ? Sema::AA_Passing : Sema::AA_Passing_CFAudited;
|
|
|
|
|
Switch the initialization required by return statements over to the
new InitializationSequence. This fixes some bugs (e.g., PR5808),
changed some diagnostics, and caused more churn than expected. What's
new:
- InitializationSequence now has a "C conversion sequence" category
and step kind, which falls back to
- Changed the diagnostics for returns to always have the result type
of the function first and the type of the expression second.
CheckSingleAssignmentConstraints to peform checking in C.
- Improved ASTs for initialization of return values. The ASTs now
capture all of the temporaries we need to create, but
intentionally do not bind the tempoary that is actually returned,
so that it won't get destroyed twice.
- Make sure to perform an (elidable!) copy of the class object that
is returned from a class.
- Fix copy elision in CodeGen to properly see through the
subexpressions that occur with elidable copies.
- Give "new" its own entity kind; as with return values and thrown
objects, we don't bind the expression so we don't call a
destructor for it.
Note that, with this patch, I've broken returning move-only types in
C++0x. We'll fix it later, when we tackle NRVO.
llvm-svn: 91669
2009-12-18 13:02:21 +08:00
|
|
|
case InitializedEntity::EK_Result:
|
|
|
|
return Sema::AA_Returning;
|
|
|
|
|
|
|
|
case InitializedEntity::EK_Temporary:
|
2013-07-12 03:13:34 +08:00
|
|
|
case InitializedEntity::EK_RelatedResult:
|
Switch the initialization required by return statements over to the
new InitializationSequence. This fixes some bugs (e.g., PR5808),
changed some diagnostics, and caused more churn than expected. What's
new:
- InitializationSequence now has a "C conversion sequence" category
and step kind, which falls back to
- Changed the diagnostics for returns to always have the result type
of the function first and the type of the expression second.
CheckSingleAssignmentConstraints to peform checking in C.
- Improved ASTs for initialization of return values. The ASTs now
capture all of the temporaries we need to create, but
intentionally do not bind the tempoary that is actually returned,
so that it won't get destroyed twice.
- Make sure to perform an (elidable!) copy of the class object that
is returned from a class.
- Fix copy elision in CodeGen to properly see through the
subexpressions that occur with elidable copies.
- Give "new" its own entity kind; as with return values and thrown
objects, we don't bind the expression so we don't call a
destructor for it.
Note that, with this patch, I've broken returning move-only types in
C++0x. We'll fix it later, when we tackle NRVO.
llvm-svn: 91669
2009-12-18 13:02:21 +08:00
|
|
|
// FIXME: Can we tell apart casting vs. converting?
|
|
|
|
return Sema::AA_Casting;
|
2011-01-27 15:10:08 +08:00
|
|
|
|
Switch the initialization required by return statements over to the
new InitializationSequence. This fixes some bugs (e.g., PR5808),
changed some diagnostics, and caused more churn than expected. What's
new:
- InitializationSequence now has a "C conversion sequence" category
and step kind, which falls back to
- Changed the diagnostics for returns to always have the result type
of the function first and the type of the expression second.
CheckSingleAssignmentConstraints to peform checking in C.
- Improved ASTs for initialization of return values. The ASTs now
capture all of the temporaries we need to create, but
intentionally do not bind the tempoary that is actually returned,
so that it won't get destroyed twice.
- Make sure to perform an (elidable!) copy of the class object that
is returned from a class.
- Fix copy elision in CodeGen to properly see through the
subexpressions that occur with elidable copies.
- Give "new" its own entity kind; as with return values and thrown
objects, we don't bind the expression so we don't call a
destructor for it.
Note that, with this patch, I've broken returning move-only types in
C++0x. We'll fix it later, when we tackle NRVO.
llvm-svn: 91669
2009-12-18 13:02:21 +08:00
|
|
|
case InitializedEntity::EK_Member:
|
2010-01-23 12:34:47 +08:00
|
|
|
case InitializedEntity::EK_ArrayElement:
|
|
|
|
case InitializedEntity::EK_VectorElement:
|
2011-09-20 07:17:44 +08:00
|
|
|
case InitializedEntity::EK_ComplexElement:
|
2010-06-08 00:14:00 +08:00
|
|
|
case InitializedEntity::EK_BlockElement:
|
2012-02-16 00:57:26 +08:00
|
|
|
case InitializedEntity::EK_LambdaCapture:
|
2013-05-07 00:48:12 +08:00
|
|
|
case InitializedEntity::EK_CompoundLiteralInit:
|
Switch the initialization required by return statements over to the
new InitializationSequence. This fixes some bugs (e.g., PR5808),
changed some diagnostics, and caused more churn than expected. What's
new:
- InitializationSequence now has a "C conversion sequence" category
and step kind, which falls back to
- Changed the diagnostics for returns to always have the result type
of the function first and the type of the expression second.
CheckSingleAssignmentConstraints to peform checking in C.
- Improved ASTs for initialization of return values. The ASTs now
capture all of the temporaries we need to create, but
intentionally do not bind the tempoary that is actually returned,
so that it won't get destroyed twice.
- Make sure to perform an (elidable!) copy of the class object that
is returned from a class.
- Fix copy elision in CodeGen to properly see through the
subexpressions that occur with elidable copies.
- Give "new" its own entity kind; as with return values and thrown
objects, we don't bind the expression so we don't call a
destructor for it.
Note that, with this patch, I've broken returning move-only types in
C++0x. We'll fix it later, when we tackle NRVO.
llvm-svn: 91669
2009-12-18 13:02:21 +08:00
|
|
|
return Sema::AA_Initializing;
|
|
|
|
}
|
|
|
|
|
2012-01-17 14:56:22 +08:00
|
|
|
llvm_unreachable("Invalid EntityKind!");
|
Switch the initialization required by return statements over to the
new InitializationSequence. This fixes some bugs (e.g., PR5808),
changed some diagnostics, and caused more churn than expected. What's
new:
- InitializationSequence now has a "C conversion sequence" category
and step kind, which falls back to
- Changed the diagnostics for returns to always have the result type
of the function first and the type of the expression second.
CheckSingleAssignmentConstraints to peform checking in C.
- Improved ASTs for initialization of return values. The ASTs now
capture all of the temporaries we need to create, but
intentionally do not bind the tempoary that is actually returned,
so that it won't get destroyed twice.
- Make sure to perform an (elidable!) copy of the class object that
is returned from a class.
- Fix copy elision in CodeGen to properly see through the
subexpressions that occur with elidable copies.
- Give "new" its own entity kind; as with return values and thrown
objects, we don't bind the expression so we don't call a
destructor for it.
Note that, with this patch, I've broken returning move-only types in
C++0x. We'll fix it later, when we tackle NRVO.
llvm-svn: 91669
2009-12-18 13:02:21 +08:00
|
|
|
}
|
|
|
|
|
2013-01-08 08:08:23 +08:00
|
|
|
/// \brief Whether we should bind a created object as a temporary when
|
2010-04-25 07:45:46 +08:00
|
|
|
/// initializing the given entity.
|
Rework our handling of copy construction of temporaries, which was a
poor (and wrong) approximation of the actual rules governing when to
build a copy and when it can be elided.
The correct implementation is actually simpler than the
approximation. When we only enumerate constructors as part of
initialization (e.g., for direct initialization or when we're copying
from a class type or one of its derived classes), we don't create a
copy. When we enumerate all conversion functions, we do create a
copy. Before, we created some extra copies and missed some
others. The new test copy-initialization.cpp shows a case where we
missed creating a (required, non-elidable) copy as part of a
user-defined conversion, which resulted in a miscompile. This commit
also fixes PR6757, where the missing copy made us reject well-formed
code in the ternary operator.
This commit also cleans up our handling of copy elision in the case
where we create an extra copy of a temporary object, which became
necessary now that we produce the right copies. The code that seeks to
find the temporary object being copied has moved into
Expr::getTemporaryObject(); it used to have two different
not-quite-the-same implementations, one in Sema and one in CodeGen.
Note that we still do not attempt to perform the named return value
optimization, so we miss copy elisions for return values and throw
expressions.
llvm-svn: 100196
2010-04-03 02:24:57 +08:00
|
|
|
static bool shouldBindAsTemporary(const InitializedEntity &Entity) {
|
Switch the initialization required by return statements over to the
new InitializationSequence. This fixes some bugs (e.g., PR5808),
changed some diagnostics, and caused more churn than expected. What's
new:
- InitializationSequence now has a "C conversion sequence" category
and step kind, which falls back to
- Changed the diagnostics for returns to always have the result type
of the function first and the type of the expression second.
CheckSingleAssignmentConstraints to peform checking in C.
- Improved ASTs for initialization of return values. The ASTs now
capture all of the temporaries we need to create, but
intentionally do not bind the tempoary that is actually returned,
so that it won't get destroyed twice.
- Make sure to perform an (elidable!) copy of the class object that
is returned from a class.
- Fix copy elision in CodeGen to properly see through the
subexpressions that occur with elidable copies.
- Give "new" its own entity kind; as with return values and thrown
objects, we don't bind the expression so we don't call a
destructor for it.
Note that, with this patch, I've broken returning move-only types in
C++0x. We'll fix it later, when we tackle NRVO.
llvm-svn: 91669
2009-12-18 13:02:21 +08:00
|
|
|
switch (Entity.getKind()) {
|
2010-01-24 08:19:41 +08:00
|
|
|
case InitializedEntity::EK_ArrayElement:
|
|
|
|
case InitializedEntity::EK_Member:
|
Rework our handling of copy construction of temporaries, which was a
poor (and wrong) approximation of the actual rules governing when to
build a copy and when it can be elided.
The correct implementation is actually simpler than the
approximation. When we only enumerate constructors as part of
initialization (e.g., for direct initialization or when we're copying
from a class type or one of its derived classes), we don't create a
copy. When we enumerate all conversion functions, we do create a
copy. Before, we created some extra copies and missed some
others. The new test copy-initialization.cpp shows a case where we
missed creating a (required, non-elidable) copy as part of a
user-defined conversion, which resulted in a miscompile. This commit
also fixes PR6757, where the missing copy made us reject well-formed
code in the ternary operator.
This commit also cleans up our handling of copy elision in the case
where we create an extra copy of a temporary object, which became
necessary now that we produce the right copies. The code that seeks to
find the temporary object being copied has moved into
Expr::getTemporaryObject(); it used to have two different
not-quite-the-same implementations, one in Sema and one in CodeGen.
Note that we still do not attempt to perform the named return value
optimization, so we miss copy elisions for return values and throw
expressions.
llvm-svn: 100196
2010-04-03 02:24:57 +08:00
|
|
|
case InitializedEntity::EK_Result:
|
Switch the initialization required by return statements over to the
new InitializationSequence. This fixes some bugs (e.g., PR5808),
changed some diagnostics, and caused more churn than expected. What's
new:
- InitializationSequence now has a "C conversion sequence" category
and step kind, which falls back to
- Changed the diagnostics for returns to always have the result type
of the function first and the type of the expression second.
CheckSingleAssignmentConstraints to peform checking in C.
- Improved ASTs for initialization of return values. The ASTs now
capture all of the temporaries we need to create, but
intentionally do not bind the tempoary that is actually returned,
so that it won't get destroyed twice.
- Make sure to perform an (elidable!) copy of the class object that
is returned from a class.
- Fix copy elision in CodeGen to properly see through the
subexpressions that occur with elidable copies.
- Give "new" its own entity kind; as with return values and thrown
objects, we don't bind the expression so we don't call a
destructor for it.
Note that, with this patch, I've broken returning move-only types in
C++0x. We'll fix it later, when we tackle NRVO.
llvm-svn: 91669
2009-12-18 13:02:21 +08:00
|
|
|
case InitializedEntity::EK_New:
|
|
|
|
case InitializedEntity::EK_Variable:
|
|
|
|
case InitializedEntity::EK_Base:
|
2011-05-01 15:04:31 +08:00
|
|
|
case InitializedEntity::EK_Delegating:
|
2010-01-23 12:34:47 +08:00
|
|
|
case InitializedEntity::EK_VectorElement:
|
2011-09-20 07:17:44 +08:00
|
|
|
case InitializedEntity::EK_ComplexElement:
|
2010-02-07 07:23:06 +08:00
|
|
|
case InitializedEntity::EK_Exception:
|
2010-06-08 00:14:00 +08:00
|
|
|
case InitializedEntity::EK_BlockElement:
|
2012-02-16 00:57:26 +08:00
|
|
|
case InitializedEntity::EK_LambdaCapture:
|
2013-05-07 00:48:12 +08:00
|
|
|
case InitializedEntity::EK_CompoundLiteralInit:
|
Switch the initialization required by return statements over to the
new InitializationSequence. This fixes some bugs (e.g., PR5808),
changed some diagnostics, and caused more churn than expected. What's
new:
- InitializationSequence now has a "C conversion sequence" category
and step kind, which falls back to
- Changed the diagnostics for returns to always have the result type
of the function first and the type of the expression second.
CheckSingleAssignmentConstraints to peform checking in C.
- Improved ASTs for initialization of return values. The ASTs now
capture all of the temporaries we need to create, but
intentionally do not bind the tempoary that is actually returned,
so that it won't get destroyed twice.
- Make sure to perform an (elidable!) copy of the class object that
is returned from a class.
- Fix copy elision in CodeGen to properly see through the
subexpressions that occur with elidable copies.
- Give "new" its own entity kind; as with return values and thrown
objects, we don't bind the expression so we don't call a
destructor for it.
Note that, with this patch, I've broken returning move-only types in
C++0x. We'll fix it later, when we tackle NRVO.
llvm-svn: 91669
2009-12-18 13:02:21 +08:00
|
|
|
return false;
|
2011-01-27 15:10:08 +08:00
|
|
|
|
Switch the initialization required by return statements over to the
new InitializationSequence. This fixes some bugs (e.g., PR5808),
changed some diagnostics, and caused more churn than expected. What's
new:
- InitializationSequence now has a "C conversion sequence" category
and step kind, which falls back to
- Changed the diagnostics for returns to always have the result type
of the function first and the type of the expression second.
CheckSingleAssignmentConstraints to peform checking in C.
- Improved ASTs for initialization of return values. The ASTs now
capture all of the temporaries we need to create, but
intentionally do not bind the tempoary that is actually returned,
so that it won't get destroyed twice.
- Make sure to perform an (elidable!) copy of the class object that
is returned from a class.
- Fix copy elision in CodeGen to properly see through the
subexpressions that occur with elidable copies.
- Give "new" its own entity kind; as with return values and thrown
objects, we don't bind the expression so we don't call a
destructor for it.
Note that, with this patch, I've broken returning move-only types in
C++0x. We'll fix it later, when we tackle NRVO.
llvm-svn: 91669
2009-12-18 13:02:21 +08:00
|
|
|
case InitializedEntity::EK_Parameter:
|
2013-08-01 02:21:45 +08:00
|
|
|
case InitializedEntity::EK_Parameter_CF_Audited:
|
Switch the initialization required by return statements over to the
new InitializationSequence. This fixes some bugs (e.g., PR5808),
changed some diagnostics, and caused more churn than expected. What's
new:
- InitializationSequence now has a "C conversion sequence" category
and step kind, which falls back to
- Changed the diagnostics for returns to always have the result type
of the function first and the type of the expression second.
CheckSingleAssignmentConstraints to peform checking in C.
- Improved ASTs for initialization of return values. The ASTs now
capture all of the temporaries we need to create, but
intentionally do not bind the tempoary that is actually returned,
so that it won't get destroyed twice.
- Make sure to perform an (elidable!) copy of the class object that
is returned from a class.
- Fix copy elision in CodeGen to properly see through the
subexpressions that occur with elidable copies.
- Give "new" its own entity kind; as with return values and thrown
objects, we don't bind the expression so we don't call a
destructor for it.
Note that, with this patch, I've broken returning move-only types in
C++0x. We'll fix it later, when we tackle NRVO.
llvm-svn: 91669
2009-12-18 13:02:21 +08:00
|
|
|
case InitializedEntity::EK_Temporary:
|
2013-07-12 00:48:06 +08:00
|
|
|
case InitializedEntity::EK_RelatedResult:
|
Switch the initialization required by return statements over to the
new InitializationSequence. This fixes some bugs (e.g., PR5808),
changed some diagnostics, and caused more churn than expected. What's
new:
- InitializationSequence now has a "C conversion sequence" category
and step kind, which falls back to
- Changed the diagnostics for returns to always have the result type
of the function first and the type of the expression second.
CheckSingleAssignmentConstraints to peform checking in C.
- Improved ASTs for initialization of return values. The ASTs now
capture all of the temporaries we need to create, but
intentionally do not bind the tempoary that is actually returned,
so that it won't get destroyed twice.
- Make sure to perform an (elidable!) copy of the class object that
is returned from a class.
- Fix copy elision in CodeGen to properly see through the
subexpressions that occur with elidable copies.
- Give "new" its own entity kind; as with return values and thrown
objects, we don't bind the expression so we don't call a
destructor for it.
Note that, with this patch, I've broken returning move-only types in
C++0x. We'll fix it later, when we tackle NRVO.
llvm-svn: 91669
2009-12-18 13:02:21 +08:00
|
|
|
return true;
|
|
|
|
}
|
2011-01-27 15:10:08 +08:00
|
|
|
|
Switch the initialization required by return statements over to the
new InitializationSequence. This fixes some bugs (e.g., PR5808),
changed some diagnostics, and caused more churn than expected. What's
new:
- InitializationSequence now has a "C conversion sequence" category
and step kind, which falls back to
- Changed the diagnostics for returns to always have the result type
of the function first and the type of the expression second.
CheckSingleAssignmentConstraints to peform checking in C.
- Improved ASTs for initialization of return values. The ASTs now
capture all of the temporaries we need to create, but
intentionally do not bind the tempoary that is actually returned,
so that it won't get destroyed twice.
- Make sure to perform an (elidable!) copy of the class object that
is returned from a class.
- Fix copy elision in CodeGen to properly see through the
subexpressions that occur with elidable copies.
- Give "new" its own entity kind; as with return values and thrown
objects, we don't bind the expression so we don't call a
destructor for it.
Note that, with this patch, I've broken returning move-only types in
C++0x. We'll fix it later, when we tackle NRVO.
llvm-svn: 91669
2009-12-18 13:02:21 +08:00
|
|
|
llvm_unreachable("missed an InitializedEntity kind?");
|
|
|
|
}
|
|
|
|
|
2010-04-25 07:45:46 +08:00
|
|
|
/// \brief Whether the given entity, when initialized with an object
|
|
|
|
/// created for that initialization, requires destruction.
|
|
|
|
static bool shouldDestroyTemporary(const InitializedEntity &Entity) {
|
|
|
|
switch (Entity.getKind()) {
|
|
|
|
case InitializedEntity::EK_Result:
|
|
|
|
case InitializedEntity::EK_New:
|
|
|
|
case InitializedEntity::EK_Base:
|
2011-05-01 15:04:31 +08:00
|
|
|
case InitializedEntity::EK_Delegating:
|
2010-04-25 07:45:46 +08:00
|
|
|
case InitializedEntity::EK_VectorElement:
|
2011-09-20 07:17:44 +08:00
|
|
|
case InitializedEntity::EK_ComplexElement:
|
2010-06-08 00:14:00 +08:00
|
|
|
case InitializedEntity::EK_BlockElement:
|
2012-02-16 00:57:26 +08:00
|
|
|
case InitializedEntity::EK_LambdaCapture:
|
2010-04-25 07:45:46 +08:00
|
|
|
return false;
|
2011-01-27 15:10:08 +08:00
|
|
|
|
2013-01-08 08:08:23 +08:00
|
|
|
case InitializedEntity::EK_Member:
|
2010-04-25 07:45:46 +08:00
|
|
|
case InitializedEntity::EK_Variable:
|
|
|
|
case InitializedEntity::EK_Parameter:
|
2013-08-01 02:21:45 +08:00
|
|
|
case InitializedEntity::EK_Parameter_CF_Audited:
|
2010-04-25 07:45:46 +08:00
|
|
|
case InitializedEntity::EK_Temporary:
|
|
|
|
case InitializedEntity::EK_ArrayElement:
|
|
|
|
case InitializedEntity::EK_Exception:
|
2013-05-07 00:48:12 +08:00
|
|
|
case InitializedEntity::EK_CompoundLiteralInit:
|
2013-07-12 00:48:06 +08:00
|
|
|
case InitializedEntity::EK_RelatedResult:
|
2010-04-25 07:45:46 +08:00
|
|
|
return true;
|
|
|
|
}
|
2011-01-27 15:10:08 +08:00
|
|
|
|
|
|
|
llvm_unreachable("missed an InitializedEntity kind?");
|
2010-04-25 07:45:46 +08:00
|
|
|
}
|
|
|
|
|
2011-10-20 00:55:56 +08:00
|
|
|
/// \brief Look for copy and move constructors and constructor templates, for
|
|
|
|
/// copying an object via direct-initialization (per C++11 [dcl.init]p16).
|
|
|
|
static void LookupCopyAndMoveConstructors(Sema &S,
|
|
|
|
OverloadCandidateSet &CandidateSet,
|
|
|
|
CXXRecordDecl *Class,
|
|
|
|
Expr *CurInitExpr) {
|
2012-12-19 08:45:41 +08:00
|
|
|
DeclContext::lookup_result R = S.LookupConstructors(Class);
|
2012-11-13 13:07:23 +08:00
|
|
|
// The container holding the constructors can under certain conditions
|
|
|
|
// be changed while iterating (e.g. because of deserialization).
|
|
|
|
// To be safe we copy the lookup results to a new container.
|
2012-12-19 08:45:41 +08:00
|
|
|
SmallVector<NamedDecl*, 16> Ctors(R.begin(), R.end());
|
2013-07-04 11:08:24 +08:00
|
|
|
for (SmallVectorImpl<NamedDecl *>::iterator
|
2012-11-13 13:07:23 +08:00
|
|
|
CI = Ctors.begin(), CE = Ctors.end(); CI != CE; ++CI) {
|
|
|
|
NamedDecl *D = *CI;
|
2011-10-20 00:55:56 +08:00
|
|
|
CXXConstructorDecl *Constructor = 0;
|
|
|
|
|
2012-11-13 13:07:23 +08:00
|
|
|
if ((Constructor = dyn_cast<CXXConstructorDecl>(D))) {
|
2011-10-20 00:55:56 +08:00
|
|
|
// Handle copy/moveconstructors, only.
|
|
|
|
if (!Constructor || Constructor->isInvalidDecl() ||
|
|
|
|
!Constructor->isCopyOrMoveConstructor() ||
|
|
|
|
!Constructor->isConvertingConstructor(/*AllowExplicit=*/true))
|
|
|
|
continue;
|
|
|
|
|
|
|
|
DeclAccessPair FoundDecl
|
|
|
|
= DeclAccessPair::make(Constructor, Constructor->getAccess());
|
|
|
|
S.AddOverloadCandidate(Constructor, FoundDecl,
|
2012-02-25 19:00:22 +08:00
|
|
|
CurInitExpr, CandidateSet);
|
2011-10-20 00:55:56 +08:00
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Handle constructor templates.
|
2012-11-13 13:07:23 +08:00
|
|
|
FunctionTemplateDecl *ConstructorTmpl = cast<FunctionTemplateDecl>(D);
|
2011-10-20 00:55:56 +08:00
|
|
|
if (ConstructorTmpl->isInvalidDecl())
|
|
|
|
continue;
|
|
|
|
|
|
|
|
Constructor = cast<CXXConstructorDecl>(
|
|
|
|
ConstructorTmpl->getTemplatedDecl());
|
|
|
|
if (!Constructor->isConvertingConstructor(/*AllowExplicit=*/true))
|
|
|
|
continue;
|
|
|
|
|
|
|
|
// FIXME: Do we need to limit this to copy-constructor-like
|
|
|
|
// candidates?
|
|
|
|
DeclAccessPair FoundDecl
|
|
|
|
= DeclAccessPair::make(ConstructorTmpl, ConstructorTmpl->getAccess());
|
|
|
|
S.AddTemplateOverloadCandidate(ConstructorTmpl, FoundDecl, 0,
|
2012-02-25 19:00:22 +08:00
|
|
|
CurInitExpr, CandidateSet, true);
|
2011-10-20 00:55:56 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/// \brief Get the location at which initialization diagnostics should appear.
|
|
|
|
static SourceLocation getInitializationLoc(const InitializedEntity &Entity,
|
|
|
|
Expr *Initializer) {
|
|
|
|
switch (Entity.getKind()) {
|
|
|
|
case InitializedEntity::EK_Result:
|
|
|
|
return Entity.getReturnLoc();
|
|
|
|
|
|
|
|
case InitializedEntity::EK_Exception:
|
|
|
|
return Entity.getThrowLoc();
|
|
|
|
|
|
|
|
case InitializedEntity::EK_Variable:
|
|
|
|
return Entity.getDecl()->getLocation();
|
|
|
|
|
2012-02-16 00:57:26 +08:00
|
|
|
case InitializedEntity::EK_LambdaCapture:
|
|
|
|
return Entity.getCaptureLoc();
|
|
|
|
|
2011-10-20 00:55:56 +08:00
|
|
|
case InitializedEntity::EK_ArrayElement:
|
|
|
|
case InitializedEntity::EK_Member:
|
|
|
|
case InitializedEntity::EK_Parameter:
|
2013-08-01 02:21:45 +08:00
|
|
|
case InitializedEntity::EK_Parameter_CF_Audited:
|
2011-10-20 00:55:56 +08:00
|
|
|
case InitializedEntity::EK_Temporary:
|
|
|
|
case InitializedEntity::EK_New:
|
|
|
|
case InitializedEntity::EK_Base:
|
|
|
|
case InitializedEntity::EK_Delegating:
|
|
|
|
case InitializedEntity::EK_VectorElement:
|
|
|
|
case InitializedEntity::EK_ComplexElement:
|
|
|
|
case InitializedEntity::EK_BlockElement:
|
2013-05-07 00:48:12 +08:00
|
|
|
case InitializedEntity::EK_CompoundLiteralInit:
|
2013-07-12 00:48:06 +08:00
|
|
|
case InitializedEntity::EK_RelatedResult:
|
2011-10-20 00:55:56 +08:00
|
|
|
return Initializer->getLocStart();
|
|
|
|
}
|
|
|
|
llvm_unreachable("missed an InitializedEntity kind?");
|
|
|
|
}
|
|
|
|
|
2010-04-18 15:40:54 +08:00
|
|
|
/// \brief Make a (potentially elidable) temporary copy of the object
|
|
|
|
/// provided by the given initializer by calling the appropriate copy
|
|
|
|
/// constructor.
|
|
|
|
///
|
|
|
|
/// \param S The Sema object used for type-checking.
|
|
|
|
///
|
2011-01-28 03:55:10 +08:00
|
|
|
/// \param T The type of the temporary object, which must either be
|
2010-04-18 15:40:54 +08:00
|
|
|
/// the type of the initializer expression or a superclass thereof.
|
|
|
|
///
|
2012-06-15 05:40:34 +08:00
|
|
|
/// \param Entity The entity being initialized.
|
2010-04-18 15:40:54 +08:00
|
|
|
///
|
|
|
|
/// \param CurInit The initializer expression.
|
|
|
|
///
|
|
|
|
/// \param IsExtraneousCopy Whether this is an "extraneous" copy that
|
|
|
|
/// is permitted in C++03 (but not C++0x) when binding a reference to
|
|
|
|
/// an rvalue.
|
|
|
|
///
|
|
|
|
/// \returns An expression that copies the initializer expression into
|
|
|
|
/// a temporary object, or an error expression if a copy could not be
|
|
|
|
/// created.
|
2010-08-24 14:29:42 +08:00
|
|
|
static ExprResult CopyObject(Sema &S,
|
2010-09-12 16:07:23 +08:00
|
|
|
QualType T,
|
|
|
|
const InitializedEntity &Entity,
|
|
|
|
ExprResult CurInit,
|
|
|
|
bool IsExtraneousCopy) {
|
2010-04-18 06:01:05 +08:00
|
|
|
// Determine which class type we're copying to.
|
2010-01-24 08:19:41 +08:00
|
|
|
Expr *CurInitExpr = (Expr *)CurInit.get();
|
2011-01-27 15:10:08 +08:00
|
|
|
CXXRecordDecl *Class = 0;
|
2010-04-18 15:40:54 +08:00
|
|
|
if (const RecordType *Record = T->getAs<RecordType>())
|
Rework our handling of copy construction of temporaries, which was a
poor (and wrong) approximation of the actual rules governing when to
build a copy and when it can be elided.
The correct implementation is actually simpler than the
approximation. When we only enumerate constructors as part of
initialization (e.g., for direct initialization or when we're copying
from a class type or one of its derived classes), we don't create a
copy. When we enumerate all conversion functions, we do create a
copy. Before, we created some extra copies and missed some
others. The new test copy-initialization.cpp shows a case where we
missed creating a (required, non-elidable) copy as part of a
user-defined conversion, which resulted in a miscompile. This commit
also fixes PR6757, where the missing copy made us reject well-formed
code in the ternary operator.
This commit also cleans up our handling of copy elision in the case
where we create an extra copy of a temporary object, which became
necessary now that we produce the right copies. The code that seeks to
find the temporary object being copied has moved into
Expr::getTemporaryObject(); it used to have two different
not-quite-the-same implementations, one in Sema and one in CodeGen.
Note that we still do not attempt to perform the named return value
optimization, so we miss copy elisions for return values and throw
expressions.
llvm-svn: 100196
2010-04-03 02:24:57 +08:00
|
|
|
Class = cast<CXXRecordDecl>(Record->getDecl());
|
|
|
|
if (!Class)
|
2012-08-24 05:35:17 +08:00
|
|
|
return CurInit;
|
Rework our handling of copy construction of temporaries, which was a
poor (and wrong) approximation of the actual rules governing when to
build a copy and when it can be elided.
The correct implementation is actually simpler than the
approximation. When we only enumerate constructors as part of
initialization (e.g., for direct initialization or when we're copying
from a class type or one of its derived classes), we don't create a
copy. When we enumerate all conversion functions, we do create a
copy. Before, we created some extra copies and missed some
others. The new test copy-initialization.cpp shows a case where we
missed creating a (required, non-elidable) copy as part of a
user-defined conversion, which resulted in a miscompile. This commit
also fixes PR6757, where the missing copy made us reject well-formed
code in the ternary operator.
This commit also cleans up our handling of copy elision in the case
where we create an extra copy of a temporary object, which became
necessary now that we produce the right copies. The code that seeks to
find the temporary object being copied has moved into
Expr::getTemporaryObject(); it used to have two different
not-quite-the-same implementations, one in Sema and one in CodeGen.
Note that we still do not attempt to perform the named return value
optimization, so we miss copy elisions for return values and throw
expressions.
llvm-svn: 100196
2010-04-03 02:24:57 +08:00
|
|
|
|
2011-01-22 02:05:27 +08:00
|
|
|
// C++0x [class.copy]p32:
|
Rework our handling of copy construction of temporaries, which was a
poor (and wrong) approximation of the actual rules governing when to
build a copy and when it can be elided.
The correct implementation is actually simpler than the
approximation. When we only enumerate constructors as part of
initialization (e.g., for direct initialization or when we're copying
from a class type or one of its derived classes), we don't create a
copy. When we enumerate all conversion functions, we do create a
copy. Before, we created some extra copies and missed some
others. The new test copy-initialization.cpp shows a case where we
missed creating a (required, non-elidable) copy as part of a
user-defined conversion, which resulted in a miscompile. This commit
also fixes PR6757, where the missing copy made us reject well-formed
code in the ternary operator.
This commit also cleans up our handling of copy elision in the case
where we create an extra copy of a temporary object, which became
necessary now that we produce the right copies. The code that seeks to
find the temporary object being copied has moved into
Expr::getTemporaryObject(); it used to have two different
not-quite-the-same implementations, one in Sema and one in CodeGen.
Note that we still do not attempt to perform the named return value
optimization, so we miss copy elisions for return values and throw
expressions.
llvm-svn: 100196
2010-04-03 02:24:57 +08:00
|
|
|
// When certain criteria are met, an implementation is allowed to
|
|
|
|
// omit the copy/move construction of a class object, even if the
|
|
|
|
// copy/move constructor and/or destructor for the object have
|
|
|
|
// side effects. [...]
|
|
|
|
// - when a temporary class object that has not been bound to a
|
|
|
|
// reference (12.2) would be copied/moved to a class object
|
|
|
|
// with the same cv-unqualified type, the copy/move operation
|
|
|
|
// can be omitted by constructing the temporary object
|
|
|
|
// directly into the target of the omitted copy/move
|
2011-01-27 15:10:08 +08:00
|
|
|
//
|
Rework our handling of copy construction of temporaries, which was a
poor (and wrong) approximation of the actual rules governing when to
build a copy and when it can be elided.
The correct implementation is actually simpler than the
approximation. When we only enumerate constructors as part of
initialization (e.g., for direct initialization or when we're copying
from a class type or one of its derived classes), we don't create a
copy. When we enumerate all conversion functions, we do create a
copy. Before, we created some extra copies and missed some
others. The new test copy-initialization.cpp shows a case where we
missed creating a (required, non-elidable) copy as part of a
user-defined conversion, which resulted in a miscompile. This commit
also fixes PR6757, where the missing copy made us reject well-formed
code in the ternary operator.
This commit also cleans up our handling of copy elision in the case
where we create an extra copy of a temporary object, which became
necessary now that we produce the right copies. The code that seeks to
find the temporary object being copied has moved into
Expr::getTemporaryObject(); it used to have two different
not-quite-the-same implementations, one in Sema and one in CodeGen.
Note that we still do not attempt to perform the named return value
optimization, so we miss copy elisions for return values and throw
expressions.
llvm-svn: 100196
2010-04-03 02:24:57 +08:00
|
|
|
// Note that the other three bullets are handled elsewhere. Copy
|
2010-05-15 08:13:29 +08:00
|
|
|
// elision for return statements and throw expressions are handled as part
|
2011-01-27 15:10:08 +08:00
|
|
|
// of constructor initialization, while copy elision for exception handlers
|
2010-05-15 08:13:29 +08:00
|
|
|
// is handled by the run-time.
|
2010-09-15 18:14:12 +08:00
|
|
|
bool Elidable = CurInitExpr->isTemporaryObject(S.Context, Class);
|
2011-10-20 00:55:56 +08:00
|
|
|
SourceLocation Loc = getInitializationLoc(Entity, CurInit.get());
|
2010-04-25 05:09:25 +08:00
|
|
|
|
2011-01-27 15:10:08 +08:00
|
|
|
// Make sure that the type we are copying is complete.
|
2012-05-05 00:32:21 +08:00
|
|
|
if (S.RequireCompleteType(Loc, T, diag::err_temp_copy_incomplete))
|
2012-08-24 05:35:17 +08:00
|
|
|
return CurInit;
|
2010-04-25 05:09:25 +08:00
|
|
|
|
2011-01-22 03:38:21 +08:00
|
|
|
// Perform overload resolution using the class's copy/move constructors.
|
2011-10-20 00:55:56 +08:00
|
|
|
// Only consider constructors and constructor templates. Per
|
|
|
|
// C++0x [dcl.init]p16, second bullet to class types, this initialization
|
|
|
|
// is direct-initialization.
|
2010-02-09 07:07:23 +08:00
|
|
|
OverloadCandidateSet CandidateSet(Loc);
|
2011-10-20 00:55:56 +08:00
|
|
|
LookupCopyAndMoveConstructors(S, CandidateSet, Class, CurInitExpr);
|
2011-01-27 15:10:08 +08:00
|
|
|
|
2011-10-05 15:56:41 +08:00
|
|
|
bool HadMultipleCandidates = (CandidateSet.size() > 1);
|
|
|
|
|
Switch the initialization required by return statements over to the
new InitializationSequence. This fixes some bugs (e.g., PR5808),
changed some diagnostics, and caused more churn than expected. What's
new:
- InitializationSequence now has a "C conversion sequence" category
and step kind, which falls back to
- Changed the diagnostics for returns to always have the result type
of the function first and the type of the expression second.
CheckSingleAssignmentConstraints to peform checking in C.
- Improved ASTs for initialization of return values. The ASTs now
capture all of the temporaries we need to create, but
intentionally do not bind the tempoary that is actually returned,
so that it won't get destroyed twice.
- Make sure to perform an (elidable!) copy of the class object that
is returned from a class.
- Fix copy elision in CodeGen to properly see through the
subexpressions that occur with elidable copies.
- Give "new" its own entity kind; as with return values and thrown
objects, we don't bind the expression so we don't call a
destructor for it.
Note that, with this patch, I've broken returning move-only types in
C++0x. We'll fix it later, when we tackle NRVO.
llvm-svn: 91669
2009-12-18 13:02:21 +08:00
|
|
|
OverloadCandidateSet::iterator Best;
|
2011-02-26 03:41:05 +08:00
|
|
|
switch (CandidateSet.BestViableFunction(S, Loc, Best)) {
|
Switch the initialization required by return statements over to the
new InitializationSequence. This fixes some bugs (e.g., PR5808),
changed some diagnostics, and caused more churn than expected. What's
new:
- InitializationSequence now has a "C conversion sequence" category
and step kind, which falls back to
- Changed the diagnostics for returns to always have the result type
of the function first and the type of the expression second.
CheckSingleAssignmentConstraints to peform checking in C.
- Improved ASTs for initialization of return values. The ASTs now
capture all of the temporaries we need to create, but
intentionally do not bind the tempoary that is actually returned,
so that it won't get destroyed twice.
- Make sure to perform an (elidable!) copy of the class object that
is returned from a class.
- Fix copy elision in CodeGen to properly see through the
subexpressions that occur with elidable copies.
- Give "new" its own entity kind; as with return values and thrown
objects, we don't bind the expression so we don't call a
destructor for it.
Note that, with this patch, I've broken returning move-only types in
C++0x. We'll fix it later, when we tackle NRVO.
llvm-svn: 91669
2009-12-18 13:02:21 +08:00
|
|
|
case OR_Success:
|
|
|
|
break;
|
2011-01-27 15:10:08 +08:00
|
|
|
|
Switch the initialization required by return statements over to the
new InitializationSequence. This fixes some bugs (e.g., PR5808),
changed some diagnostics, and caused more churn than expected. What's
new:
- InitializationSequence now has a "C conversion sequence" category
and step kind, which falls back to
- Changed the diagnostics for returns to always have the result type
of the function first and the type of the expression second.
CheckSingleAssignmentConstraints to peform checking in C.
- Improved ASTs for initialization of return values. The ASTs now
capture all of the temporaries we need to create, but
intentionally do not bind the tempoary that is actually returned,
so that it won't get destroyed twice.
- Make sure to perform an (elidable!) copy of the class object that
is returned from a class.
- Fix copy elision in CodeGen to properly see through the
subexpressions that occur with elidable copies.
- Give "new" its own entity kind; as with return values and thrown
objects, we don't bind the expression so we don't call a
destructor for it.
Note that, with this patch, I've broken returning move-only types in
C++0x. We'll fix it later, when we tackle NRVO.
llvm-svn: 91669
2009-12-18 13:02:21 +08:00
|
|
|
case OR_No_Viable_Function:
|
2010-06-07 23:58:05 +08:00
|
|
|
S.Diag(Loc, IsExtraneousCopy && !S.isSFINAEContext()
|
|
|
|
? diag::ext_rvalue_to_reference_temp_copy_no_viable
|
|
|
|
: diag::err_temp_copy_no_viable)
|
2009-12-19 11:01:41 +08:00
|
|
|
<< (int)Entity.getKind() << CurInitExpr->getType()
|
Switch the initialization required by return statements over to the
new InitializationSequence. This fixes some bugs (e.g., PR5808),
changed some diagnostics, and caused more churn than expected. What's
new:
- InitializationSequence now has a "C conversion sequence" category
and step kind, which falls back to
- Changed the diagnostics for returns to always have the result type
of the function first and the type of the expression second.
CheckSingleAssignmentConstraints to peform checking in C.
- Improved ASTs for initialization of return values. The ASTs now
capture all of the temporaries we need to create, but
intentionally do not bind the tempoary that is actually returned,
so that it won't get destroyed twice.
- Make sure to perform an (elidable!) copy of the class object that
is returned from a class.
- Fix copy elision in CodeGen to properly see through the
subexpressions that occur with elidable copies.
- Give "new" its own entity kind; as with return values and thrown
objects, we don't bind the expression so we don't call a
destructor for it.
Note that, with this patch, I've broken returning move-only types in
C++0x. We'll fix it later, when we tackle NRVO.
llvm-svn: 91669
2009-12-18 13:02:21 +08:00
|
|
|
<< CurInitExpr->getSourceRange();
|
2012-02-25 19:00:22 +08:00
|
|
|
CandidateSet.NoteCandidates(S, OCD_AllCandidates, CurInitExpr);
|
2010-06-07 23:58:05 +08:00
|
|
|
if (!IsExtraneousCopy || S.isSFINAEContext())
|
2010-08-27 07:41:50 +08:00
|
|
|
return ExprError();
|
2012-08-24 05:35:17 +08:00
|
|
|
return CurInit;
|
2011-01-27 15:10:08 +08:00
|
|
|
|
Switch the initialization required by return statements over to the
new InitializationSequence. This fixes some bugs (e.g., PR5808),
changed some diagnostics, and caused more churn than expected. What's
new:
- InitializationSequence now has a "C conversion sequence" category
and step kind, which falls back to
- Changed the diagnostics for returns to always have the result type
of the function first and the type of the expression second.
CheckSingleAssignmentConstraints to peform checking in C.
- Improved ASTs for initialization of return values. The ASTs now
capture all of the temporaries we need to create, but
intentionally do not bind the tempoary that is actually returned,
so that it won't get destroyed twice.
- Make sure to perform an (elidable!) copy of the class object that
is returned from a class.
- Fix copy elision in CodeGen to properly see through the
subexpressions that occur with elidable copies.
- Give "new" its own entity kind; as with return values and thrown
objects, we don't bind the expression so we don't call a
destructor for it.
Note that, with this patch, I've broken returning move-only types in
C++0x. We'll fix it later, when we tackle NRVO.
llvm-svn: 91669
2009-12-18 13:02:21 +08:00
|
|
|
case OR_Ambiguous:
|
|
|
|
S.Diag(Loc, diag::err_temp_copy_ambiguous)
|
2009-12-19 11:01:41 +08:00
|
|
|
<< (int)Entity.getKind() << CurInitExpr->getType()
|
Switch the initialization required by return statements over to the
new InitializationSequence. This fixes some bugs (e.g., PR5808),
changed some diagnostics, and caused more churn than expected. What's
new:
- InitializationSequence now has a "C conversion sequence" category
and step kind, which falls back to
- Changed the diagnostics for returns to always have the result type
of the function first and the type of the expression second.
CheckSingleAssignmentConstraints to peform checking in C.
- Improved ASTs for initialization of return values. The ASTs now
capture all of the temporaries we need to create, but
intentionally do not bind the tempoary that is actually returned,
so that it won't get destroyed twice.
- Make sure to perform an (elidable!) copy of the class object that
is returned from a class.
- Fix copy elision in CodeGen to properly see through the
subexpressions that occur with elidable copies.
- Give "new" its own entity kind; as with return values and thrown
objects, we don't bind the expression so we don't call a
destructor for it.
Note that, with this patch, I've broken returning move-only types in
C++0x. We'll fix it later, when we tackle NRVO.
llvm-svn: 91669
2009-12-18 13:02:21 +08:00
|
|
|
<< CurInitExpr->getSourceRange();
|
2012-02-25 19:00:22 +08:00
|
|
|
CandidateSet.NoteCandidates(S, OCD_ViableCandidates, CurInitExpr);
|
2010-08-27 07:41:50 +08:00
|
|
|
return ExprError();
|
2011-01-27 15:10:08 +08:00
|
|
|
|
Switch the initialization required by return statements over to the
new InitializationSequence. This fixes some bugs (e.g., PR5808),
changed some diagnostics, and caused more churn than expected. What's
new:
- InitializationSequence now has a "C conversion sequence" category
and step kind, which falls back to
- Changed the diagnostics for returns to always have the result type
of the function first and the type of the expression second.
CheckSingleAssignmentConstraints to peform checking in C.
- Improved ASTs for initialization of return values. The ASTs now
capture all of the temporaries we need to create, but
intentionally do not bind the tempoary that is actually returned,
so that it won't get destroyed twice.
- Make sure to perform an (elidable!) copy of the class object that
is returned from a class.
- Fix copy elision in CodeGen to properly see through the
subexpressions that occur with elidable copies.
- Give "new" its own entity kind; as with return values and thrown
objects, we don't bind the expression so we don't call a
destructor for it.
Note that, with this patch, I've broken returning move-only types in
C++0x. We'll fix it later, when we tackle NRVO.
llvm-svn: 91669
2009-12-18 13:02:21 +08:00
|
|
|
case OR_Deleted:
|
|
|
|
S.Diag(Loc, diag::err_temp_copy_deleted)
|
2009-12-19 11:01:41 +08:00
|
|
|
<< (int)Entity.getKind() << CurInitExpr->getType()
|
Switch the initialization required by return statements over to the
new InitializationSequence. This fixes some bugs (e.g., PR5808),
changed some diagnostics, and caused more churn than expected. What's
new:
- InitializationSequence now has a "C conversion sequence" category
and step kind, which falls back to
- Changed the diagnostics for returns to always have the result type
of the function first and the type of the expression second.
CheckSingleAssignmentConstraints to peform checking in C.
- Improved ASTs for initialization of return values. The ASTs now
capture all of the temporaries we need to create, but
intentionally do not bind the tempoary that is actually returned,
so that it won't get destroyed twice.
- Make sure to perform an (elidable!) copy of the class object that
is returned from a class.
- Fix copy elision in CodeGen to properly see through the
subexpressions that occur with elidable copies.
- Give "new" its own entity kind; as with return values and thrown
objects, we don't bind the expression so we don't call a
destructor for it.
Note that, with this patch, I've broken returning move-only types in
C++0x. We'll fix it later, when we tackle NRVO.
llvm-svn: 91669
2009-12-18 13:02:21 +08:00
|
|
|
<< CurInitExpr->getSourceRange();
|
2012-03-31 04:53:28 +08:00
|
|
|
S.NoteDeletedFunction(Best->Function);
|
2010-08-27 07:41:50 +08:00
|
|
|
return ExprError();
|
Switch the initialization required by return statements over to the
new InitializationSequence. This fixes some bugs (e.g., PR5808),
changed some diagnostics, and caused more churn than expected. What's
new:
- InitializationSequence now has a "C conversion sequence" category
and step kind, which falls back to
- Changed the diagnostics for returns to always have the result type
of the function first and the type of the expression second.
CheckSingleAssignmentConstraints to peform checking in C.
- Improved ASTs for initialization of return values. The ASTs now
capture all of the temporaries we need to create, but
intentionally do not bind the tempoary that is actually returned,
so that it won't get destroyed twice.
- Make sure to perform an (elidable!) copy of the class object that
is returned from a class.
- Fix copy elision in CodeGen to properly see through the
subexpressions that occur with elidable copies.
- Give "new" its own entity kind; as with return values and thrown
objects, we don't bind the expression so we don't call a
destructor for it.
Note that, with this patch, I've broken returning move-only types in
C++0x. We'll fix it later, when we tackle NRVO.
llvm-svn: 91669
2009-12-18 13:02:21 +08:00
|
|
|
}
|
|
|
|
|
2010-04-18 06:01:05 +08:00
|
|
|
CXXConstructorDecl *Constructor = cast<CXXConstructorDecl>(Best->Function);
|
2012-08-24 06:51:59 +08:00
|
|
|
SmallVector<Expr*, 8> ConstructorArgs;
|
2010-04-18 06:01:05 +08:00
|
|
|
CurInit.release(); // Ownership transferred into MultiExprArg, below.
|
2010-04-18 15:40:54 +08:00
|
|
|
|
2010-04-22 02:47:17 +08:00
|
|
|
S.CheckConstructorAccess(Loc, Constructor, Entity,
|
2010-06-07 23:58:05 +08:00
|
|
|
Best->FoundDecl.getAccess(), IsExtraneousCopy);
|
2010-04-18 15:40:54 +08:00
|
|
|
|
|
|
|
if (IsExtraneousCopy) {
|
|
|
|
// If this is a totally extraneous copy for C++03 reference
|
|
|
|
// binding purposes, just return the original initialization
|
2010-04-18 15:57:34 +08:00
|
|
|
// expression. We don't generate an (elided) copy operation here
|
|
|
|
// because doing so would require us to pass down a flag to avoid
|
|
|
|
// infinite recursion, where each step adds another extraneous,
|
|
|
|
// elidable copy.
|
|
|
|
|
|
|
|
// Instantiate the default arguments of any extra parameters in
|
|
|
|
// the selected copy constructor, as if we were going to create a
|
|
|
|
// proper call to the copy constructor.
|
|
|
|
for (unsigned I = 1, N = Constructor->getNumParams(); I != N; ++I) {
|
|
|
|
ParmVarDecl *Parm = Constructor->getParamDecl(I);
|
|
|
|
if (S.RequireCompleteType(Loc, Parm->getType(),
|
2012-05-05 00:32:21 +08:00
|
|
|
diag::err_call_incomplete_argument))
|
2010-04-18 15:57:34 +08:00
|
|
|
break;
|
|
|
|
|
|
|
|
// Build the default argument expression; we don't actually care
|
|
|
|
// if this succeeds or not, because this routine will complain
|
|
|
|
// if there was a problem.
|
|
|
|
S.BuildCXXDefaultArgExpr(Loc, Constructor, Parm);
|
|
|
|
}
|
2010-04-18 15:40:54 +08:00
|
|
|
|
|
|
|
return S.Owned(CurInitExpr);
|
|
|
|
}
|
2011-01-27 15:10:08 +08:00
|
|
|
|
2010-04-18 06:01:05 +08:00
|
|
|
// Determine the arguments required to actually perform the
|
2010-04-18 15:40:54 +08:00
|
|
|
// constructor call (we might have derived-to-base conversions, or
|
|
|
|
// the copy constructor may have default arguments).
|
2013-05-03 23:05:50 +08:00
|
|
|
if (S.CompleteConstructorCall(Constructor, CurInitExpr, Loc, ConstructorArgs))
|
2010-08-27 07:41:50 +08:00
|
|
|
return ExprError();
|
2010-04-18 06:01:05 +08:00
|
|
|
|
2010-04-25 08:55:24 +08:00
|
|
|
// Actually perform the constructor call.
|
|
|
|
CurInit = S.BuildCXXConstructExpr(Loc, T, Constructor, Elidable,
|
2012-08-24 05:35:17 +08:00
|
|
|
ConstructorArgs,
|
2011-10-05 15:56:41 +08:00
|
|
|
HadMultipleCandidates,
|
2012-12-19 09:39:02 +08:00
|
|
|
/*ListInit*/ false,
|
2010-08-24 15:32:53 +08:00
|
|
|
/*ZeroInit*/ false,
|
2010-10-25 16:47:36 +08:00
|
|
|
CXXConstructExpr::CK_Complete,
|
|
|
|
SourceRange());
|
2011-01-27 15:10:08 +08:00
|
|
|
|
2010-04-25 08:55:24 +08:00
|
|
|
// If we're supposed to bind temporaries, do so.
|
|
|
|
if (!CurInit.isInvalid() && shouldBindAsTemporary(Entity))
|
|
|
|
CurInit = S.MaybeBindToTemporary(CurInit.takeAs<Expr>());
|
2012-08-24 05:35:17 +08:00
|
|
|
return CurInit;
|
Switch the initialization required by return statements over to the
new InitializationSequence. This fixes some bugs (e.g., PR5808),
changed some diagnostics, and caused more churn than expected. What's
new:
- InitializationSequence now has a "C conversion sequence" category
and step kind, which falls back to
- Changed the diagnostics for returns to always have the result type
of the function first and the type of the expression second.
CheckSingleAssignmentConstraints to peform checking in C.
- Improved ASTs for initialization of return values. The ASTs now
capture all of the temporaries we need to create, but
intentionally do not bind the tempoary that is actually returned,
so that it won't get destroyed twice.
- Make sure to perform an (elidable!) copy of the class object that
is returned from a class.
- Fix copy elision in CodeGen to properly see through the
subexpressions that occur with elidable copies.
- Give "new" its own entity kind; as with return values and thrown
objects, we don't bind the expression so we don't call a
destructor for it.
Note that, with this patch, I've broken returning move-only types in
C++0x. We'll fix it later, when we tackle NRVO.
llvm-svn: 91669
2009-12-18 13:02:21 +08:00
|
|
|
}
|
2009-12-10 07:02:17 +08:00
|
|
|
|
2011-10-20 00:55:56 +08:00
|
|
|
/// \brief Check whether elidable copy construction for binding a reference to
|
|
|
|
/// a temporary would have succeeded if we were building in C++98 mode, for
|
|
|
|
/// -Wc++98-compat.
|
|
|
|
static void CheckCXX98CompatAccessibleCopy(Sema &S,
|
|
|
|
const InitializedEntity &Entity,
|
|
|
|
Expr *CurInitExpr) {
|
2013-01-02 19:42:31 +08:00
|
|
|
assert(S.getLangOpts().CPlusPlus11);
|
2011-10-20 00:55:56 +08:00
|
|
|
|
|
|
|
const RecordType *Record = CurInitExpr->getType()->getAs<RecordType>();
|
|
|
|
if (!Record)
|
|
|
|
return;
|
|
|
|
|
|
|
|
SourceLocation Loc = getInitializationLoc(Entity, CurInitExpr);
|
|
|
|
if (S.Diags.getDiagnosticLevel(diag::warn_cxx98_compat_temp_copy, Loc)
|
|
|
|
== DiagnosticsEngine::Ignored)
|
|
|
|
return;
|
|
|
|
|
|
|
|
// Find constructors which would have been considered.
|
|
|
|
OverloadCandidateSet CandidateSet(Loc);
|
|
|
|
LookupCopyAndMoveConstructors(
|
|
|
|
S, CandidateSet, cast<CXXRecordDecl>(Record->getDecl()), CurInitExpr);
|
|
|
|
|
|
|
|
// Perform overload resolution.
|
|
|
|
OverloadCandidateSet::iterator Best;
|
|
|
|
OverloadingResult OR = CandidateSet.BestViableFunction(S, Loc, Best);
|
|
|
|
|
|
|
|
PartialDiagnostic Diag = S.PDiag(diag::warn_cxx98_compat_temp_copy)
|
|
|
|
<< OR << (int)Entity.getKind() << CurInitExpr->getType()
|
|
|
|
<< CurInitExpr->getSourceRange();
|
|
|
|
|
|
|
|
switch (OR) {
|
|
|
|
case OR_Success:
|
|
|
|
S.CheckConstructorAccess(Loc, cast<CXXConstructorDecl>(Best->Function),
|
Fix several problems with protected access control:
- The [class.protected] restriction is non-trivial for any instance
member, even if the access lacks an object (for example, if it's
a pointer-to-member constant). In this case, it is equivalent to
requiring the naming class to equal the context class.
- The [class.protected] restriction applies to accesses to constructors
and destructors. A protected constructor or destructor can only be
used to create or destroy a base subobject, as a direct result.
- Several places were dropping or misapplying object information.
The standard could really be much clearer about what the object type is
supposed to be in some of these accesses. Usually it's easy enough to
find a reasonable answer, but still, the standard makes a very confident
statement about accesses to instance members only being possible in
either pointer-to-member literals or member access expressions, which
just completely ignores concepts like constructor and destructor
calls, using declarations, unevaluated field references, etc.
llvm-svn: 154248
2012-04-07 11:04:20 +08:00
|
|
|
Entity, Best->FoundDecl.getAccess(), Diag);
|
2011-10-20 00:55:56 +08:00
|
|
|
// FIXME: Check default arguments as far as that's possible.
|
|
|
|
break;
|
|
|
|
|
|
|
|
case OR_No_Viable_Function:
|
|
|
|
S.Diag(Loc, Diag);
|
2012-02-25 19:00:22 +08:00
|
|
|
CandidateSet.NoteCandidates(S, OCD_AllCandidates, CurInitExpr);
|
2011-10-20 00:55:56 +08:00
|
|
|
break;
|
|
|
|
|
|
|
|
case OR_Ambiguous:
|
|
|
|
S.Diag(Loc, Diag);
|
2012-02-25 19:00:22 +08:00
|
|
|
CandidateSet.NoteCandidates(S, OCD_ViableCandidates, CurInitExpr);
|
2011-10-20 00:55:56 +08:00
|
|
|
break;
|
|
|
|
|
|
|
|
case OR_Deleted:
|
|
|
|
S.Diag(Loc, Diag);
|
2012-03-31 04:53:28 +08:00
|
|
|
S.NoteDeletedFunction(Best->Function);
|
2011-10-20 00:55:56 +08:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2010-04-22 08:20:18 +08:00
|
|
|
void InitializationSequence::PrintInitLocationNote(Sema &S,
|
|
|
|
const InitializedEntity &Entity) {
|
2013-08-01 02:21:45 +08:00
|
|
|
if (Entity.isParameterKind() && Entity.getDecl()) {
|
2010-04-22 08:20:18 +08:00
|
|
|
if (Entity.getDecl()->getLocation().isInvalid())
|
|
|
|
return;
|
|
|
|
|
|
|
|
if (Entity.getDecl()->getDeclName())
|
|
|
|
S.Diag(Entity.getDecl()->getLocation(), diag::note_parameter_named_here)
|
|
|
|
<< Entity.getDecl()->getDeclName();
|
|
|
|
else
|
|
|
|
S.Diag(Entity.getDecl()->getLocation(), diag::note_parameter_here);
|
|
|
|
}
|
2013-07-12 00:48:06 +08:00
|
|
|
else if (Entity.getKind() == InitializedEntity::EK_RelatedResult &&
|
|
|
|
Entity.getMethodDecl())
|
|
|
|
S.Diag(Entity.getMethodDecl()->getLocation(),
|
|
|
|
diag::note_method_return_type_change)
|
|
|
|
<< Entity.getMethodDecl()->getDeclName();
|
2010-04-22 08:20:18 +08:00
|
|
|
}
|
|
|
|
|
2011-07-15 03:07:55 +08:00
|
|
|
static bool isReferenceBinding(const InitializationSequence::Step &s) {
|
|
|
|
return s.Kind == InitializationSequence::SK_BindReference ||
|
|
|
|
s.Kind == InitializationSequence::SK_BindReferenceToTemporary;
|
|
|
|
}
|
|
|
|
|
2013-05-07 00:48:12 +08:00
|
|
|
/// Returns true if the parameters describe a constructor initialization of
|
|
|
|
/// an explicit temporary object, e.g. "Point(x, y)".
|
|
|
|
static bool isExplicitTemporary(const InitializedEntity &Entity,
|
|
|
|
const InitializationKind &Kind,
|
|
|
|
unsigned NumArgs) {
|
|
|
|
switch (Entity.getKind()) {
|
|
|
|
case InitializedEntity::EK_Temporary:
|
|
|
|
case InitializedEntity::EK_CompoundLiteralInit:
|
2013-07-12 00:48:06 +08:00
|
|
|
case InitializedEntity::EK_RelatedResult:
|
2013-05-07 00:48:12 +08:00
|
|
|
break;
|
|
|
|
default:
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
switch (Kind.getKind()) {
|
|
|
|
case InitializationKind::IK_DirectList:
|
|
|
|
return true;
|
|
|
|
// FIXME: Hack to work around cast weirdness.
|
|
|
|
case InitializationKind::IK_Direct:
|
|
|
|
case InitializationKind::IK_Value:
|
|
|
|
return NumArgs != 1;
|
|
|
|
default:
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-12-22 22:44:04 +08:00
|
|
|
static ExprResult
|
|
|
|
PerformConstructorInitialization(Sema &S,
|
|
|
|
const InitializedEntity &Entity,
|
|
|
|
const InitializationKind &Kind,
|
|
|
|
MultiExprArg Args,
|
|
|
|
const InitializationSequence::Step& Step,
|
2012-12-19 09:39:02 +08:00
|
|
|
bool &ConstructorInitRequiresZeroInit,
|
2013-09-07 13:49:53 +08:00
|
|
|
bool IsListInitialization,
|
|
|
|
SourceLocation LBraceLoc,
|
|
|
|
SourceLocation RBraceLoc) {
|
2011-12-22 22:44:04 +08:00
|
|
|
unsigned NumArgs = Args.size();
|
|
|
|
CXXConstructorDecl *Constructor
|
|
|
|
= cast<CXXConstructorDecl>(Step.Function.Function);
|
|
|
|
bool HadMultipleCandidates = Step.Function.HadMultipleCandidates;
|
|
|
|
|
|
|
|
// Build a call to the selected constructor.
|
2012-08-24 06:51:59 +08:00
|
|
|
SmallVector<Expr*, 8> ConstructorArgs;
|
2011-12-22 22:44:04 +08:00
|
|
|
SourceLocation Loc = (Kind.isCopyInit() && Kind.getEqualLoc().isValid())
|
|
|
|
? Kind.getEqualLoc()
|
|
|
|
: Kind.getLocation();
|
|
|
|
|
|
|
|
if (Kind.getKind() == InitializationKind::IK_Default) {
|
|
|
|
// Force even a trivial, implicit default constructor to be
|
|
|
|
// semantically checked. We do this explicitly because we don't build
|
|
|
|
// the definition for completely trivial constructors.
|
2012-02-24 16:37:56 +08:00
|
|
|
assert(Constructor->getParent() && "No parent class for constructor.");
|
2011-12-22 22:44:04 +08:00
|
|
|
if (Constructor->isDefaulted() && Constructor->isDefaultConstructor() &&
|
2012-02-24 15:48:37 +08:00
|
|
|
Constructor->isTrivial() && !Constructor->isUsed(false))
|
2011-12-22 22:44:04 +08:00
|
|
|
S.DefineImplicitDefaultConstructor(Loc, Constructor);
|
|
|
|
}
|
|
|
|
|
|
|
|
ExprResult CurInit = S.Owned((Expr *)0);
|
|
|
|
|
2012-02-25 07:56:31 +08:00
|
|
|
// C++ [over.match.copy]p1:
|
|
|
|
// - When initializing a temporary to be bound to the first parameter
|
|
|
|
// of a constructor that takes a reference to possibly cv-qualified
|
|
|
|
// T as its first argument, called with a single argument in the
|
|
|
|
// context of direct-initialization, explicit conversion functions
|
|
|
|
// are also considered.
|
|
|
|
bool AllowExplicitConv = Kind.AllowExplicit() && !Kind.isCopyInit() &&
|
|
|
|
Args.size() == 1 &&
|
|
|
|
Constructor->isCopyOrMoveConstructor();
|
|
|
|
|
2011-12-22 22:44:04 +08:00
|
|
|
// Determine the arguments required to actually perform the constructor
|
|
|
|
// call.
|
2012-08-24 05:35:17 +08:00
|
|
|
if (S.CompleteConstructorCall(Constructor, Args,
|
2012-02-25 07:56:31 +08:00
|
|
|
Loc, ConstructorArgs,
|
2013-02-05 13:52:24 +08:00
|
|
|
AllowExplicitConv,
|
|
|
|
IsListInitialization))
|
2011-12-22 22:44:04 +08:00
|
|
|
return ExprError();
|
|
|
|
|
|
|
|
|
2013-05-07 00:48:12 +08:00
|
|
|
if (isExplicitTemporary(Entity, Kind, NumArgs)) {
|
2011-12-22 22:44:04 +08:00
|
|
|
// An explicitly-constructed temporary, e.g., X(1, 2).
|
2012-02-02 11:46:19 +08:00
|
|
|
S.MarkFunctionReferenced(Loc, Constructor);
|
2013-05-04 14:44:46 +08:00
|
|
|
if (S.DiagnoseUseOfDecl(Constructor, Loc))
|
|
|
|
return ExprError();
|
2011-12-22 22:44:04 +08:00
|
|
|
|
|
|
|
TypeSourceInfo *TSInfo = Entity.getTypeSourceInfo();
|
|
|
|
if (!TSInfo)
|
|
|
|
TSInfo = S.Context.getTrivialTypeSourceInfo(Entity.getType(), Loc);
|
2013-09-07 13:49:53 +08:00
|
|
|
SourceRange ParenOrBraceRange =
|
|
|
|
(Kind.getKind() == InitializationKind::IK_DirectList)
|
|
|
|
? SourceRange(LBraceLoc, RBraceLoc)
|
|
|
|
: Kind.getParenRange();
|
2011-12-22 22:44:04 +08:00
|
|
|
|
2012-12-19 09:39:02 +08:00
|
|
|
CurInit = S.Owned(
|
|
|
|
new (S.Context) CXXTemporaryObjectExpr(S.Context, Constructor,
|
|
|
|
TSInfo, ConstructorArgs,
|
2013-09-07 13:49:53 +08:00
|
|
|
ParenOrBraceRange,
|
2012-12-19 09:39:02 +08:00
|
|
|
HadMultipleCandidates,
|
2013-09-07 19:22:02 +08:00
|
|
|
IsListInitialization,
|
2012-12-19 09:39:02 +08:00
|
|
|
ConstructorInitRequiresZeroInit));
|
2011-12-22 22:44:04 +08:00
|
|
|
} else {
|
|
|
|
CXXConstructExpr::ConstructionKind ConstructKind =
|
|
|
|
CXXConstructExpr::CK_Complete;
|
|
|
|
|
|
|
|
if (Entity.getKind() == InitializedEntity::EK_Base) {
|
|
|
|
ConstructKind = Entity.getBaseSpecifier()->isVirtual() ?
|
|
|
|
CXXConstructExpr::CK_VirtualBase :
|
|
|
|
CXXConstructExpr::CK_NonVirtualBase;
|
|
|
|
} else if (Entity.getKind() == InitializedEntity::EK_Delegating) {
|
|
|
|
ConstructKind = CXXConstructExpr::CK_Delegating;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Only get the parenthesis range if it is a direct construction.
|
|
|
|
SourceRange parenRange =
|
|
|
|
Kind.getKind() == InitializationKind::IK_Direct ?
|
|
|
|
Kind.getParenRange() : SourceRange();
|
|
|
|
|
|
|
|
// If the entity allows NRVO, mark the construction as elidable
|
|
|
|
// unconditionally.
|
|
|
|
if (Entity.allowsNRVO())
|
|
|
|
CurInit = S.BuildCXXConstructExpr(Loc, Entity.getType(),
|
|
|
|
Constructor, /*Elidable=*/true,
|
2012-08-24 05:35:17 +08:00
|
|
|
ConstructorArgs,
|
2011-12-22 22:44:04 +08:00
|
|
|
HadMultipleCandidates,
|
2012-12-19 09:39:02 +08:00
|
|
|
IsListInitialization,
|
2011-12-22 22:44:04 +08:00
|
|
|
ConstructorInitRequiresZeroInit,
|
|
|
|
ConstructKind,
|
|
|
|
parenRange);
|
|
|
|
else
|
|
|
|
CurInit = S.BuildCXXConstructExpr(Loc, Entity.getType(),
|
|
|
|
Constructor,
|
2012-08-24 05:35:17 +08:00
|
|
|
ConstructorArgs,
|
2011-12-22 22:44:04 +08:00
|
|
|
HadMultipleCandidates,
|
2012-12-19 09:39:02 +08:00
|
|
|
IsListInitialization,
|
2011-12-22 22:44:04 +08:00
|
|
|
ConstructorInitRequiresZeroInit,
|
|
|
|
ConstructKind,
|
|
|
|
parenRange);
|
|
|
|
}
|
|
|
|
if (CurInit.isInvalid())
|
|
|
|
return ExprError();
|
|
|
|
|
|
|
|
// Only check access if all of that succeeded.
|
|
|
|
S.CheckConstructorAccess(Loc, Constructor, Entity,
|
|
|
|
Step.Function.FoundDecl.getAccess());
|
2013-05-04 14:44:46 +08:00
|
|
|
if (S.DiagnoseUseOfDecl(Step.Function.FoundDecl, Loc))
|
|
|
|
return ExprError();
|
2011-12-22 22:44:04 +08:00
|
|
|
|
|
|
|
if (shouldBindAsTemporary(Entity))
|
2013-06-13 06:31:48 +08:00
|
|
|
CurInit = S.MaybeBindToTemporary(CurInit.take());
|
2011-12-22 22:44:04 +08:00
|
|
|
|
2012-08-24 05:35:17 +08:00
|
|
|
return CurInit;
|
2011-12-22 22:44:04 +08:00
|
|
|
}
|
|
|
|
|
2012-06-05 06:27:30 +08:00
|
|
|
/// Determine whether the specified InitializedEntity definitely has a lifetime
|
|
|
|
/// longer than the current full-expression. Conservatively returns false if
|
|
|
|
/// it's unclear.
|
|
|
|
static bool
|
|
|
|
InitializedEntityOutlivesFullExpression(const InitializedEntity &Entity) {
|
|
|
|
const InitializedEntity *Top = &Entity;
|
|
|
|
while (Top->getParent())
|
|
|
|
Top = Top->getParent();
|
|
|
|
|
|
|
|
switch (Top->getKind()) {
|
|
|
|
case InitializedEntity::EK_Variable:
|
|
|
|
case InitializedEntity::EK_Result:
|
|
|
|
case InitializedEntity::EK_Exception:
|
|
|
|
case InitializedEntity::EK_Member:
|
|
|
|
case InitializedEntity::EK_New:
|
|
|
|
case InitializedEntity::EK_Base:
|
|
|
|
case InitializedEntity::EK_Delegating:
|
|
|
|
return true;
|
|
|
|
|
|
|
|
case InitializedEntity::EK_ArrayElement:
|
|
|
|
case InitializedEntity::EK_VectorElement:
|
|
|
|
case InitializedEntity::EK_BlockElement:
|
|
|
|
case InitializedEntity::EK_ComplexElement:
|
|
|
|
// Could not determine what the full initialization is. Assume it might not
|
|
|
|
// outlive the full-expression.
|
|
|
|
return false;
|
|
|
|
|
|
|
|
case InitializedEntity::EK_Parameter:
|
2013-08-01 02:21:45 +08:00
|
|
|
case InitializedEntity::EK_Parameter_CF_Audited:
|
2012-06-05 06:27:30 +08:00
|
|
|
case InitializedEntity::EK_Temporary:
|
|
|
|
case InitializedEntity::EK_LambdaCapture:
|
2013-05-07 00:48:12 +08:00
|
|
|
case InitializedEntity::EK_CompoundLiteralInit:
|
2013-07-12 00:48:06 +08:00
|
|
|
case InitializedEntity::EK_RelatedResult:
|
2012-06-05 06:27:30 +08:00
|
|
|
// The entity being initialized might not outlive the full-expression.
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
llvm_unreachable("unknown entity kind");
|
|
|
|
}
|
|
|
|
|
2013-06-05 08:46:14 +08:00
|
|
|
/// Determine the declaration which an initialized entity ultimately refers to,
|
|
|
|
/// for the purpose of lifetime-extending a temporary bound to a reference in
|
|
|
|
/// the initialization of \p Entity.
|
|
|
|
static const ValueDecl *
|
|
|
|
getDeclForTemporaryLifetimeExtension(const InitializedEntity &Entity,
|
|
|
|
const ValueDecl *FallbackDecl = 0) {
|
|
|
|
// C++11 [class.temporary]p5:
|
|
|
|
switch (Entity.getKind()) {
|
|
|
|
case InitializedEntity::EK_Variable:
|
|
|
|
// The temporary [...] persists for the lifetime of the reference
|
|
|
|
return Entity.getDecl();
|
|
|
|
|
|
|
|
case InitializedEntity::EK_Member:
|
|
|
|
// For subobjects, we look at the complete object.
|
|
|
|
if (Entity.getParent())
|
|
|
|
return getDeclForTemporaryLifetimeExtension(*Entity.getParent(),
|
|
|
|
Entity.getDecl());
|
|
|
|
|
|
|
|
// except:
|
|
|
|
// -- A temporary bound to a reference member in a constructor's
|
|
|
|
// ctor-initializer persists until the constructor exits.
|
|
|
|
return Entity.getDecl();
|
|
|
|
|
|
|
|
case InitializedEntity::EK_Parameter:
|
2013-08-01 02:21:45 +08:00
|
|
|
case InitializedEntity::EK_Parameter_CF_Audited:
|
2013-06-05 08:46:14 +08:00
|
|
|
// -- A temporary bound to a reference parameter in a function call
|
|
|
|
// persists until the completion of the full-expression containing
|
|
|
|
// the call.
|
|
|
|
case InitializedEntity::EK_Result:
|
|
|
|
// -- The lifetime of a temporary bound to the returned value in a
|
|
|
|
// function return statement is not extended; the temporary is
|
|
|
|
// destroyed at the end of the full-expression in the return statement.
|
|
|
|
case InitializedEntity::EK_New:
|
|
|
|
// -- A temporary bound to a reference in a new-initializer persists
|
|
|
|
// until the completion of the full-expression containing the
|
|
|
|
// new-initializer.
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
case InitializedEntity::EK_Temporary:
|
|
|
|
case InitializedEntity::EK_CompoundLiteralInit:
|
2013-07-12 00:48:06 +08:00
|
|
|
case InitializedEntity::EK_RelatedResult:
|
2013-06-05 08:46:14 +08:00
|
|
|
// We don't yet know the storage duration of the surrounding temporary.
|
|
|
|
// Assume it's got full-expression duration for now, it will patch up our
|
|
|
|
// storage duration if that's not correct.
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
case InitializedEntity::EK_ArrayElement:
|
|
|
|
// For subobjects, we look at the complete object.
|
|
|
|
return getDeclForTemporaryLifetimeExtension(*Entity.getParent(),
|
|
|
|
FallbackDecl);
|
|
|
|
|
|
|
|
case InitializedEntity::EK_Base:
|
|
|
|
case InitializedEntity::EK_Delegating:
|
|
|
|
// We can reach this case for aggregate initialization in a constructor:
|
|
|
|
// struct A { int &&r; };
|
|
|
|
// struct B : A { B() : A{0} {} };
|
|
|
|
// In this case, use the innermost field decl as the context.
|
|
|
|
return FallbackDecl;
|
|
|
|
|
|
|
|
case InitializedEntity::EK_BlockElement:
|
|
|
|
case InitializedEntity::EK_LambdaCapture:
|
|
|
|
case InitializedEntity::EK_Exception:
|
|
|
|
case InitializedEntity::EK_VectorElement:
|
|
|
|
case InitializedEntity::EK_ComplexElement:
|
2013-06-15 08:30:29 +08:00
|
|
|
return 0;
|
2013-06-05 08:46:14 +08:00
|
|
|
}
|
2013-06-05 23:37:50 +08:00
|
|
|
llvm_unreachable("unknown entity kind");
|
2013-06-05 08:46:14 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
static void performLifetimeExtension(Expr *Init, const ValueDecl *ExtendingD);
|
|
|
|
|
|
|
|
/// Update a glvalue expression that is used as the initializer of a reference
|
|
|
|
/// to note that its lifetime is extended.
|
2013-06-15 08:30:29 +08:00
|
|
|
/// \return \c true if any temporary had its lifetime extended.
|
|
|
|
static bool performReferenceExtension(Expr *Init, const ValueDecl *ExtendingD) {
|
2013-06-05 08:46:14 +08:00
|
|
|
if (InitListExpr *ILE = dyn_cast<InitListExpr>(Init)) {
|
|
|
|
if (ILE->getNumInits() == 1 && ILE->isGLValue()) {
|
|
|
|
// This is just redundant braces around an initializer. Step over it.
|
|
|
|
Init = ILE->getInit(0);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-06-15 08:30:29 +08:00
|
|
|
// Walk past any constructs which we can lifetime-extend across.
|
|
|
|
Expr *Old;
|
|
|
|
do {
|
|
|
|
Old = Init;
|
|
|
|
|
|
|
|
// Step over any subobject adjustments; we may have a materialized
|
|
|
|
// temporary inside them.
|
|
|
|
SmallVector<const Expr *, 2> CommaLHSs;
|
|
|
|
SmallVector<SubobjectAdjustment, 2> Adjustments;
|
|
|
|
Init = const_cast<Expr *>(
|
|
|
|
Init->skipRValueSubobjectAdjustments(CommaLHSs, Adjustments));
|
|
|
|
|
|
|
|
// Per current approach for DR1376, look through casts to reference type
|
|
|
|
// when performing lifetime extension.
|
|
|
|
if (CastExpr *CE = dyn_cast<CastExpr>(Init))
|
|
|
|
if (CE->getSubExpr()->isGLValue())
|
|
|
|
Init = CE->getSubExpr();
|
|
|
|
|
|
|
|
// FIXME: Per DR1213, subscripting on an array temporary produces an xvalue.
|
|
|
|
// It's unclear if binding a reference to that xvalue extends the array
|
|
|
|
// temporary.
|
|
|
|
} while (Init != Old);
|
|
|
|
|
2013-06-05 08:46:14 +08:00
|
|
|
if (MaterializeTemporaryExpr *ME = dyn_cast<MaterializeTemporaryExpr>(Init)) {
|
|
|
|
// Update the storage duration of the materialized temporary.
|
|
|
|
// FIXME: Rebuild the expression instead of mutating it.
|
|
|
|
ME->setExtendingDecl(ExtendingD);
|
|
|
|
performLifetimeExtension(ME->GetTemporaryExpr(), ExtendingD);
|
2013-06-15 08:30:29 +08:00
|
|
|
return true;
|
2013-06-05 08:46:14 +08:00
|
|
|
}
|
2013-06-15 08:30:29 +08:00
|
|
|
|
|
|
|
return false;
|
2013-06-05 08:46:14 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
/// Update a prvalue expression that is going to be materialized as a
|
|
|
|
/// lifetime-extended temporary.
|
|
|
|
static void performLifetimeExtension(Expr *Init, const ValueDecl *ExtendingD) {
|
|
|
|
// Dig out the expression which constructs the extended temporary.
|
|
|
|
SmallVector<const Expr *, 2> CommaLHSs;
|
|
|
|
SmallVector<SubobjectAdjustment, 2> Adjustments;
|
|
|
|
Init = const_cast<Expr *>(
|
|
|
|
Init->skipRValueSubobjectAdjustments(CommaLHSs, Adjustments));
|
|
|
|
|
2013-06-13 04:42:33 +08:00
|
|
|
if (CXXBindTemporaryExpr *BTE = dyn_cast<CXXBindTemporaryExpr>(Init))
|
|
|
|
Init = BTE->getSubExpr();
|
|
|
|
|
2013-06-13 06:31:48 +08:00
|
|
|
if (CXXStdInitializerListExpr *ILE =
|
2013-06-15 08:30:29 +08:00
|
|
|
dyn_cast<CXXStdInitializerListExpr>(Init)) {
|
|
|
|
performReferenceExtension(ILE->getSubExpr(), ExtendingD);
|
|
|
|
return;
|
|
|
|
}
|
2013-06-13 06:31:48 +08:00
|
|
|
|
2013-06-05 08:46:14 +08:00
|
|
|
if (InitListExpr *ILE = dyn_cast<InitListExpr>(Init)) {
|
2013-06-13 06:31:48 +08:00
|
|
|
if (ILE->getType()->isArrayType()) {
|
2013-06-05 08:46:14 +08:00
|
|
|
for (unsigned I = 0, N = ILE->getNumInits(); I != N; ++I)
|
|
|
|
performLifetimeExtension(ILE->getInit(I), ExtendingD);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2013-06-13 06:31:48 +08:00
|
|
|
if (CXXRecordDecl *RD = ILE->getType()->getAsCXXRecordDecl()) {
|
2013-06-05 08:46:14 +08:00
|
|
|
assert(RD->isAggregate() && "aggregate init on non-aggregate");
|
|
|
|
|
|
|
|
// If we lifetime-extend a braced initializer which is initializing an
|
|
|
|
// aggregate, and that aggregate contains reference members which are
|
|
|
|
// bound to temporaries, those temporaries are also lifetime-extended.
|
|
|
|
if (RD->isUnion() && ILE->getInitializedFieldInUnion() &&
|
|
|
|
ILE->getInitializedFieldInUnion()->getType()->isReferenceType())
|
|
|
|
performReferenceExtension(ILE->getInit(0), ExtendingD);
|
|
|
|
else {
|
|
|
|
unsigned Index = 0;
|
|
|
|
for (RecordDecl::field_iterator I = RD->field_begin(),
|
|
|
|
E = RD->field_end();
|
|
|
|
I != E; ++I) {
|
2013-07-01 14:08:20 +08:00
|
|
|
if (Index >= ILE->getNumInits())
|
|
|
|
break;
|
2013-06-05 08:46:14 +08:00
|
|
|
if (I->isUnnamedBitfield())
|
|
|
|
continue;
|
2013-06-28 06:54:33 +08:00
|
|
|
Expr *SubInit = ILE->getInit(Index);
|
2013-06-05 08:46:14 +08:00
|
|
|
if (I->getType()->isReferenceType())
|
2013-06-28 06:54:33 +08:00
|
|
|
performReferenceExtension(SubInit, ExtendingD);
|
|
|
|
else if (isa<InitListExpr>(SubInit) ||
|
|
|
|
isa<CXXStdInitializerListExpr>(SubInit))
|
2013-06-05 08:46:14 +08:00
|
|
|
// This may be either aggregate-initialization of a member or
|
|
|
|
// initialization of a std::initializer_list object. Either way,
|
|
|
|
// we should recursively lifetime-extend that initializer.
|
2013-06-28 06:54:33 +08:00
|
|
|
performLifetimeExtension(SubInit, ExtendingD);
|
2013-06-05 08:46:14 +08:00
|
|
|
++Index;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-06-13 06:31:48 +08:00
|
|
|
static void warnOnLifetimeExtension(Sema &S, const InitializedEntity &Entity,
|
|
|
|
const Expr *Init, bool IsInitializerList,
|
|
|
|
const ValueDecl *ExtendingDecl) {
|
|
|
|
// Warn if a field lifetime-extends a temporary.
|
|
|
|
if (isa<FieldDecl>(ExtendingDecl)) {
|
|
|
|
if (IsInitializerList) {
|
|
|
|
S.Diag(Init->getExprLoc(), diag::warn_dangling_std_initializer_list)
|
|
|
|
<< /*at end of constructor*/true;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool IsSubobjectMember = false;
|
|
|
|
for (const InitializedEntity *Ent = Entity.getParent(); Ent;
|
|
|
|
Ent = Ent->getParent()) {
|
|
|
|
if (Ent->getKind() != InitializedEntity::EK_Base) {
|
|
|
|
IsSubobjectMember = true;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
S.Diag(Init->getExprLoc(),
|
|
|
|
diag::warn_bind_ref_member_to_temporary)
|
|
|
|
<< ExtendingDecl << Init->getSourceRange()
|
|
|
|
<< IsSubobjectMember << IsInitializerList;
|
|
|
|
if (IsSubobjectMember)
|
|
|
|
S.Diag(ExtendingDecl->getLocation(),
|
|
|
|
diag::note_ref_subobject_of_member_declared_here);
|
|
|
|
else
|
|
|
|
S.Diag(ExtendingDecl->getLocation(),
|
|
|
|
diag::note_ref_or_ptr_member_declared_here)
|
|
|
|
<< /*is pointer*/false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-09-22 05:19:19 +08:00
|
|
|
static void DiagnoseNarrowingInInitList(Sema &S,
|
|
|
|
const ImplicitConversionSequence &ICS,
|
|
|
|
QualType PreNarrowingType,
|
|
|
|
QualType EntityType,
|
|
|
|
const Expr *PostInit);
|
|
|
|
|
2011-01-27 15:10:08 +08:00
|
|
|
ExprResult
|
2009-12-10 07:02:17 +08:00
|
|
|
InitializationSequence::Perform(Sema &S,
|
|
|
|
const InitializedEntity &Entity,
|
|
|
|
const InitializationKind &Kind,
|
2010-08-27 07:41:50 +08:00
|
|
|
MultiExprArg Args,
|
2009-12-11 01:56:55 +08:00
|
|
|
QualType *ResultType) {
|
2011-06-05 21:59:05 +08:00
|
|
|
if (Failed()) {
|
2013-05-03 23:05:50 +08:00
|
|
|
Diagnose(S, Entity, Kind, Args);
|
2010-08-27 07:41:50 +08:00
|
|
|
return ExprError();
|
2009-12-10 07:02:17 +08:00
|
|
|
}
|
2011-01-27 15:10:08 +08:00
|
|
|
|
2011-06-05 21:59:11 +08:00
|
|
|
if (getKind() == DependentSequence) {
|
2009-12-11 01:56:55 +08:00
|
|
|
// If the declaration is a non-dependent, incomplete array type
|
|
|
|
// that has an initializer, then its type will be completed once
|
|
|
|
// the initializer is instantiated.
|
2009-12-22 23:35:07 +08:00
|
|
|
if (ResultType && !Entity.getType()->isDependentType() &&
|
2009-12-11 01:56:55 +08:00
|
|
|
Args.size() == 1) {
|
2009-12-22 23:35:07 +08:00
|
|
|
QualType DeclType = Entity.getType();
|
2009-12-11 01:56:55 +08:00
|
|
|
if (const IncompleteArrayType *ArrayT
|
|
|
|
= S.Context.getAsIncompleteArrayType(DeclType)) {
|
|
|
|
// FIXME: We don't currently have the ability to accurately
|
|
|
|
// compute the length of an initializer list without
|
|
|
|
// performing full type-checking of the initializer list
|
|
|
|
// (since we have to determine where braces are implicitly
|
|
|
|
// introduced and such). So, we fall back to making the array
|
|
|
|
// type a dependently-sized array type with no specified
|
|
|
|
// bound.
|
2012-08-24 05:35:17 +08:00
|
|
|
if (isa<InitListExpr>((Expr *)Args[0])) {
|
2009-12-11 01:56:55 +08:00
|
|
|
SourceRange Brackets;
|
2009-12-22 23:35:07 +08:00
|
|
|
|
2009-12-11 01:56:55 +08:00
|
|
|
// Scavange the location of the brackets from the entity, if we can.
|
2009-12-22 23:35:07 +08:00
|
|
|
if (DeclaratorDecl *DD = Entity.getDecl()) {
|
|
|
|
if (TypeSourceInfo *TInfo = DD->getTypeSourceInfo()) {
|
|
|
|
TypeLoc TL = TInfo->getTypeLoc();
|
2013-02-19 06:06:02 +08:00
|
|
|
if (IncompleteArrayTypeLoc ArrayLoc =
|
|
|
|
TL.getAs<IncompleteArrayTypeLoc>())
|
|
|
|
Brackets = ArrayLoc.getBracketsRange();
|
2009-12-22 23:35:07 +08:00
|
|
|
}
|
2009-12-11 01:56:55 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
*ResultType
|
|
|
|
= S.Context.getDependentSizedArrayType(ArrayT->getElementType(),
|
|
|
|
/*NumElts=*/0,
|
|
|
|
ArrayT->getSizeModifier(),
|
|
|
|
ArrayT->getIndexTypeCVRQualifiers(),
|
|
|
|
Brackets);
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
}
|
Represent C++ direct initializers as ParenListExprs before semantic analysis
instead of having a special-purpose function.
- ActOnCXXDirectInitializer, which was mostly duplication of
AddInitializerToDecl (leading e.g. to PR10620, which Eli fixed a few days
ago), is dropped completely.
- MultiInitializer, which was an ugly hack I added, is dropped again.
- We now have the infrastructure in place to distinguish between
int x = {1};
int x({1});
int x{1};
-- VarDecl now has getInitStyle(), which indicates which of the above was used.
-- CXXConstructExpr now has a flag to indicate that it represents list-
initialization, although this is not yet used.
- InstantiateInitializer was renamed to SubstInitializer and simplified.
- ActOnParenOrParenListExpr has been replaced by ActOnParenListExpr, which
always produces a ParenListExpr. Placed that so far failed to convert that
back to a ParenExpr containing comma operators have been fixed. I'm pretty
sure I could have made a crashing test case before this.
The end result is a (I hope) considerably cleaner design of initializers.
More importantly, the fact that I can now distinguish between the various
initialization kinds means that I can get the tricky generalized initializer
test cases Johannes Schaub supplied to work. (This is not yet done.)
This commit passed self-host, with the resulting compiler passing the tests. I
hope it doesn't break more complicated code. It's a pretty big change, but one
that I feel is necessary.
llvm-svn: 150318
2012-02-12 07:51:47 +08:00
|
|
|
if (Kind.getKind() == InitializationKind::IK_Direct &&
|
|
|
|
!Kind.isExplicitCast()) {
|
|
|
|
// Rebuild the ParenListExpr.
|
|
|
|
SourceRange ParenRange = Kind.getParenRange();
|
|
|
|
return S.ActOnParenListExpr(ParenRange.getBegin(), ParenRange.getEnd(),
|
2012-08-24 05:35:17 +08:00
|
|
|
Args);
|
Represent C++ direct initializers as ParenListExprs before semantic analysis
instead of having a special-purpose function.
- ActOnCXXDirectInitializer, which was mostly duplication of
AddInitializerToDecl (leading e.g. to PR10620, which Eli fixed a few days
ago), is dropped completely.
- MultiInitializer, which was an ugly hack I added, is dropped again.
- We now have the infrastructure in place to distinguish between
int x = {1};
int x({1});
int x{1};
-- VarDecl now has getInitStyle(), which indicates which of the above was used.
-- CXXConstructExpr now has a flag to indicate that it represents list-
initialization, although this is not yet used.
- InstantiateInitializer was renamed to SubstInitializer and simplified.
- ActOnParenOrParenListExpr has been replaced by ActOnParenListExpr, which
always produces a ParenListExpr. Placed that so far failed to convert that
back to a ParenExpr containing comma operators have been fixed. I'm pretty
sure I could have made a crashing test case before this.
The end result is a (I hope) considerably cleaner design of initializers.
More importantly, the fact that I can now distinguish between the various
initialization kinds means that I can get the tricky generalized initializer
test cases Johannes Schaub supplied to work. (This is not yet done.)
This commit passed self-host, with the resulting compiler passing the tests. I
hope it doesn't break more complicated code. It's a pretty big change, but one
that I feel is necessary.
llvm-svn: 150318
2012-02-12 07:51:47 +08:00
|
|
|
}
|
2011-06-23 04:02:16 +08:00
|
|
|
assert(Kind.getKind() == InitializationKind::IK_Copy ||
|
2012-04-04 12:06:51 +08:00
|
|
|
Kind.isExplicitCast() ||
|
|
|
|
Kind.getKind() == InitializationKind::IK_DirectList);
|
2012-08-24 05:35:17 +08:00
|
|
|
return ExprResult(Args[0]);
|
2009-12-10 07:02:17 +08:00
|
|
|
}
|
|
|
|
|
2011-06-05 21:59:11 +08:00
|
|
|
// No steps means no initialization.
|
|
|
|
if (Steps.empty())
|
2009-12-16 09:38:02 +08:00
|
|
|
return S.Owned((Expr *)0);
|
2011-01-27 15:10:08 +08:00
|
|
|
|
2013-01-02 19:42:31 +08:00
|
|
|
if (S.getLangOpts().CPlusPlus11 && Entity.getType()->isReferenceType() &&
|
2012-08-24 05:35:17 +08:00
|
|
|
Args.size() == 1 && isa<InitListExpr>(Args[0]) &&
|
2013-08-01 02:21:45 +08:00
|
|
|
!Entity.isParameterKind()) {
|
2012-04-19 14:58:00 +08:00
|
|
|
// Produce a C++98 compatibility warning if we are initializing a reference
|
|
|
|
// from an initializer list. For parameters, we produce a better warning
|
|
|
|
// elsewhere.
|
2012-08-24 05:35:17 +08:00
|
|
|
Expr *Init = Args[0];
|
2012-04-19 14:58:00 +08:00
|
|
|
S.Diag(Init->getLocStart(), diag::warn_cxx98_compat_reference_list_init)
|
|
|
|
<< Init->getSourceRange();
|
2012-06-05 06:27:30 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
// Diagnose cases where we initialize a pointer to an array temporary, and the
|
|
|
|
// pointer obviously outlives the temporary.
|
2012-08-24 05:35:17 +08:00
|
|
|
if (Args.size() == 1 && Args[0]->getType()->isArrayType() &&
|
2012-06-05 06:27:30 +08:00
|
|
|
Entity.getType()->isPointerType() &&
|
|
|
|
InitializedEntityOutlivesFullExpression(Entity)) {
|
2012-08-24 05:35:17 +08:00
|
|
|
Expr *Init = Args[0];
|
2012-06-05 06:27:30 +08:00
|
|
|
Expr::LValueClassification Kind = Init->ClassifyLValue(S.Context);
|
|
|
|
if (Kind == Expr::LV_ClassTemporary || Kind == Expr::LV_ArrayTemporary)
|
|
|
|
S.Diag(Init->getLocStart(), diag::warn_temporary_array_to_pointer_decay)
|
|
|
|
<< Init->getSourceRange();
|
2012-04-19 14:58:00 +08:00
|
|
|
}
|
|
|
|
|
2009-12-22 23:35:07 +08:00
|
|
|
QualType DestType = Entity.getType().getNonReferenceType();
|
|
|
|
// FIXME: Ugly hack around the fact that Entity.getType() is not
|
2009-12-22 10:10:53 +08:00
|
|
|
// the same as Entity.getDecl()->getType() in cases involving type merging,
|
|
|
|
// and we want latter when it makes sense.
|
2009-12-11 01:56:55 +08:00
|
|
|
if (ResultType)
|
2009-12-22 10:10:53 +08:00
|
|
|
*ResultType = Entity.getDecl() ? Entity.getDecl()->getType() :
|
2009-12-22 23:35:07 +08:00
|
|
|
Entity.getType();
|
2009-12-10 07:02:17 +08:00
|
|
|
|
2010-08-24 14:29:42 +08:00
|
|
|
ExprResult CurInit = S.Owned((Expr *)0);
|
2011-01-27 15:10:08 +08:00
|
|
|
|
|
|
|
// For initialization steps that start with a single initializer,
|
2009-12-16 09:38:02 +08:00
|
|
|
// grab the only argument out the Args and place it into the "current"
|
|
|
|
// initializer.
|
|
|
|
switch (Steps.front().Kind) {
|
Switch the initialization required by return statements over to the
new InitializationSequence. This fixes some bugs (e.g., PR5808),
changed some diagnostics, and caused more churn than expected. What's
new:
- InitializationSequence now has a "C conversion sequence" category
and step kind, which falls back to
- Changed the diagnostics for returns to always have the result type
of the function first and the type of the expression second.
CheckSingleAssignmentConstraints to peform checking in C.
- Improved ASTs for initialization of return values. The ASTs now
capture all of the temporaries we need to create, but
intentionally do not bind the tempoary that is actually returned,
so that it won't get destroyed twice.
- Make sure to perform an (elidable!) copy of the class object that
is returned from a class.
- Fix copy elision in CodeGen to properly see through the
subexpressions that occur with elidable copies.
- Give "new" its own entity kind; as with return values and thrown
objects, we don't bind the expression so we don't call a
destructor for it.
Note that, with this patch, I've broken returning move-only types in
C++0x. We'll fix it later, when we tackle NRVO.
llvm-svn: 91669
2009-12-18 13:02:21 +08:00
|
|
|
case SK_ResolveAddressOfOverloadedFunction:
|
|
|
|
case SK_CastDerivedToBaseRValue:
|
2010-07-20 12:20:21 +08:00
|
|
|
case SK_CastDerivedToBaseXValue:
|
Switch the initialization required by return statements over to the
new InitializationSequence. This fixes some bugs (e.g., PR5808),
changed some diagnostics, and caused more churn than expected. What's
new:
- InitializationSequence now has a "C conversion sequence" category
and step kind, which falls back to
- Changed the diagnostics for returns to always have the result type
of the function first and the type of the expression second.
CheckSingleAssignmentConstraints to peform checking in C.
- Improved ASTs for initialization of return values. The ASTs now
capture all of the temporaries we need to create, but
intentionally do not bind the tempoary that is actually returned,
so that it won't get destroyed twice.
- Make sure to perform an (elidable!) copy of the class object that
is returned from a class.
- Fix copy elision in CodeGen to properly see through the
subexpressions that occur with elidable copies.
- Give "new" its own entity kind; as with return values and thrown
objects, we don't bind the expression so we don't call a
destructor for it.
Note that, with this patch, I've broken returning move-only types in
C++0x. We'll fix it later, when we tackle NRVO.
llvm-svn: 91669
2009-12-18 13:02:21 +08:00
|
|
|
case SK_CastDerivedToBaseLValue:
|
|
|
|
case SK_BindReference:
|
|
|
|
case SK_BindReferenceToTemporary:
|
2010-04-18 15:40:54 +08:00
|
|
|
case SK_ExtraneousCopyToTemporary:
|
Switch the initialization required by return statements over to the
new InitializationSequence. This fixes some bugs (e.g., PR5808),
changed some diagnostics, and caused more churn than expected. What's
new:
- InitializationSequence now has a "C conversion sequence" category
and step kind, which falls back to
- Changed the diagnostics for returns to always have the result type
of the function first and the type of the expression second.
CheckSingleAssignmentConstraints to peform checking in C.
- Improved ASTs for initialization of return values. The ASTs now
capture all of the temporaries we need to create, but
intentionally do not bind the tempoary that is actually returned,
so that it won't get destroyed twice.
- Make sure to perform an (elidable!) copy of the class object that
is returned from a class.
- Fix copy elision in CodeGen to properly see through the
subexpressions that occur with elidable copies.
- Give "new" its own entity kind; as with return values and thrown
objects, we don't bind the expression so we don't call a
destructor for it.
Note that, with this patch, I've broken returning move-only types in
C++0x. We'll fix it later, when we tackle NRVO.
llvm-svn: 91669
2009-12-18 13:02:21 +08:00
|
|
|
case SK_UserConversion:
|
|
|
|
case SK_QualificationConversionLValue:
|
2010-07-20 12:20:21 +08:00
|
|
|
case SK_QualificationConversionXValue:
|
Switch the initialization required by return statements over to the
new InitializationSequence. This fixes some bugs (e.g., PR5808),
changed some diagnostics, and caused more churn than expected. What's
new:
- InitializationSequence now has a "C conversion sequence" category
and step kind, which falls back to
- Changed the diagnostics for returns to always have the result type
of the function first and the type of the expression second.
CheckSingleAssignmentConstraints to peform checking in C.
- Improved ASTs for initialization of return values. The ASTs now
capture all of the temporaries we need to create, but
intentionally do not bind the tempoary that is actually returned,
so that it won't get destroyed twice.
- Make sure to perform an (elidable!) copy of the class object that
is returned from a class.
- Fix copy elision in CodeGen to properly see through the
subexpressions that occur with elidable copies.
- Give "new" its own entity kind; as with return values and thrown
objects, we don't bind the expression so we don't call a
destructor for it.
Note that, with this patch, I've broken returning move-only types in
C++0x. We'll fix it later, when we tackle NRVO.
llvm-svn: 91669
2009-12-18 13:02:21 +08:00
|
|
|
case SK_QualificationConversionRValue:
|
2013-04-11 08:58:58 +08:00
|
|
|
case SK_LValueToRValue:
|
Switch the initialization required by return statements over to the
new InitializationSequence. This fixes some bugs (e.g., PR5808),
changed some diagnostics, and caused more churn than expected. What's
new:
- InitializationSequence now has a "C conversion sequence" category
and step kind, which falls back to
- Changed the diagnostics for returns to always have the result type
of the function first and the type of the expression second.
CheckSingleAssignmentConstraints to peform checking in C.
- Improved ASTs for initialization of return values. The ASTs now
capture all of the temporaries we need to create, but
intentionally do not bind the tempoary that is actually returned,
so that it won't get destroyed twice.
- Make sure to perform an (elidable!) copy of the class object that
is returned from a class.
- Fix copy elision in CodeGen to properly see through the
subexpressions that occur with elidable copies.
- Give "new" its own entity kind; as with return values and thrown
objects, we don't bind the expression so we don't call a
destructor for it.
Note that, with this patch, I've broken returning move-only types in
C++0x. We'll fix it later, when we tackle NRVO.
llvm-svn: 91669
2009-12-18 13:02:21 +08:00
|
|
|
case SK_ConversionSequence:
|
2013-09-22 05:19:19 +08:00
|
|
|
case SK_ConversionSequenceNoNarrowing:
|
Switch the initialization required by return statements over to the
new InitializationSequence. This fixes some bugs (e.g., PR5808),
changed some diagnostics, and caused more churn than expected. What's
new:
- InitializationSequence now has a "C conversion sequence" category
and step kind, which falls back to
- Changed the diagnostics for returns to always have the result type
of the function first and the type of the expression second.
CheckSingleAssignmentConstraints to peform checking in C.
- Improved ASTs for initialization of return values. The ASTs now
capture all of the temporaries we need to create, but
intentionally do not bind the tempoary that is actually returned,
so that it won't get destroyed twice.
- Make sure to perform an (elidable!) copy of the class object that
is returned from a class.
- Fix copy elision in CodeGen to properly see through the
subexpressions that occur with elidable copies.
- Give "new" its own entity kind; as with return values and thrown
objects, we don't bind the expression so we don't call a
destructor for it.
Note that, with this patch, I've broken returning move-only types in
C++0x. We'll fix it later, when we tackle NRVO.
llvm-svn: 91669
2009-12-18 13:02:21 +08:00
|
|
|
case SK_ListInitialization:
|
2011-11-28 00:50:07 +08:00
|
|
|
case SK_UnwrapInitList:
|
|
|
|
case SK_RewrapInitList:
|
Switch the initialization required by return statements over to the
new InitializationSequence. This fixes some bugs (e.g., PR5808),
changed some diagnostics, and caused more churn than expected. What's
new:
- InitializationSequence now has a "C conversion sequence" category
and step kind, which falls back to
- Changed the diagnostics for returns to always have the result type
of the function first and the type of the expression second.
CheckSingleAssignmentConstraints to peform checking in C.
- Improved ASTs for initialization of return values. The ASTs now
capture all of the temporaries we need to create, but
intentionally do not bind the tempoary that is actually returned,
so that it won't get destroyed twice.
- Make sure to perform an (elidable!) copy of the class object that
is returned from a class.
- Fix copy elision in CodeGen to properly see through the
subexpressions that occur with elidable copies.
- Give "new" its own entity kind; as with return values and thrown
objects, we don't bind the expression so we don't call a
destructor for it.
Note that, with this patch, I've broken returning move-only types in
C++0x. We'll fix it later, when we tackle NRVO.
llvm-svn: 91669
2009-12-18 13:02:21 +08:00
|
|
|
case SK_CAssignment:
|
2009-12-19 16:11:05 +08:00
|
|
|
case SK_StringInit:
|
2011-02-23 02:29:51 +08:00
|
|
|
case SK_ObjCObjectConversion:
|
2011-06-16 07:02:42 +08:00
|
|
|
case SK_ArrayInit:
|
2012-02-16 06:38:09 +08:00
|
|
|
case SK_ParenthesizedArrayInit:
|
2011-06-16 07:02:42 +08:00
|
|
|
case SK_PassByIndirectCopyRestore:
|
|
|
|
case SK_PassByIndirectRestore:
|
2012-01-18 06:49:42 +08:00
|
|
|
case SK_ProduceObjCObject:
|
2013-01-20 20:31:11 +08:00
|
|
|
case SK_StdInitializerList:
|
2013-02-07 18:55:47 +08:00
|
|
|
case SK_OCLSamplerInit:
|
2013-01-20 20:31:11 +08:00
|
|
|
case SK_OCLZeroEvent: {
|
Switch the initialization required by return statements over to the
new InitializationSequence. This fixes some bugs (e.g., PR5808),
changed some diagnostics, and caused more churn than expected. What's
new:
- InitializationSequence now has a "C conversion sequence" category
and step kind, which falls back to
- Changed the diagnostics for returns to always have the result type
of the function first and the type of the expression second.
CheckSingleAssignmentConstraints to peform checking in C.
- Improved ASTs for initialization of return values. The ASTs now
capture all of the temporaries we need to create, but
intentionally do not bind the tempoary that is actually returned,
so that it won't get destroyed twice.
- Make sure to perform an (elidable!) copy of the class object that
is returned from a class.
- Fix copy elision in CodeGen to properly see through the
subexpressions that occur with elidable copies.
- Give "new" its own entity kind; as with return values and thrown
objects, we don't bind the expression so we don't call a
destructor for it.
Note that, with this patch, I've broken returning move-only types in
C++0x. We'll fix it later, when we tackle NRVO.
llvm-svn: 91669
2009-12-18 13:02:21 +08:00
|
|
|
assert(Args.size() == 1);
|
2012-08-24 05:35:17 +08:00
|
|
|
CurInit = Args[0];
|
2011-04-09 02:41:53 +08:00
|
|
|
if (!CurInit.get()) return ExprError();
|
Switch the initialization required by return statements over to the
new InitializationSequence. This fixes some bugs (e.g., PR5808),
changed some diagnostics, and caused more churn than expected. What's
new:
- InitializationSequence now has a "C conversion sequence" category
and step kind, which falls back to
- Changed the diagnostics for returns to always have the result type
of the function first and the type of the expression second.
CheckSingleAssignmentConstraints to peform checking in C.
- Improved ASTs for initialization of return values. The ASTs now
capture all of the temporaries we need to create, but
intentionally do not bind the tempoary that is actually returned,
so that it won't get destroyed twice.
- Make sure to perform an (elidable!) copy of the class object that
is returned from a class.
- Fix copy elision in CodeGen to properly see through the
subexpressions that occur with elidable copies.
- Give "new" its own entity kind; as with return values and thrown
objects, we don't bind the expression so we don't call a
destructor for it.
Note that, with this patch, I've broken returning move-only types in
C++0x. We'll fix it later, when we tackle NRVO.
llvm-svn: 91669
2009-12-18 13:02:21 +08:00
|
|
|
break;
|
2010-12-04 11:47:34 +08:00
|
|
|
}
|
2011-01-27 15:10:08 +08:00
|
|
|
|
Switch the initialization required by return statements over to the
new InitializationSequence. This fixes some bugs (e.g., PR5808),
changed some diagnostics, and caused more churn than expected. What's
new:
- InitializationSequence now has a "C conversion sequence" category
and step kind, which falls back to
- Changed the diagnostics for returns to always have the result type
of the function first and the type of the expression second.
CheckSingleAssignmentConstraints to peform checking in C.
- Improved ASTs for initialization of return values. The ASTs now
capture all of the temporaries we need to create, but
intentionally do not bind the tempoary that is actually returned,
so that it won't get destroyed twice.
- Make sure to perform an (elidable!) copy of the class object that
is returned from a class.
- Fix copy elision in CodeGen to properly see through the
subexpressions that occur with elidable copies.
- Give "new" its own entity kind; as with return values and thrown
objects, we don't bind the expression so we don't call a
destructor for it.
Note that, with this patch, I've broken returning move-only types in
C++0x. We'll fix it later, when we tackle NRVO.
llvm-svn: 91669
2009-12-18 13:02:21 +08:00
|
|
|
case SK_ConstructorInitialization:
|
2012-07-05 16:39:21 +08:00
|
|
|
case SK_ListConstructorCall:
|
Switch the initialization required by return statements over to the
new InitializationSequence. This fixes some bugs (e.g., PR5808),
changed some diagnostics, and caused more churn than expected. What's
new:
- InitializationSequence now has a "C conversion sequence" category
and step kind, which falls back to
- Changed the diagnostics for returns to always have the result type
of the function first and the type of the expression second.
CheckSingleAssignmentConstraints to peform checking in C.
- Improved ASTs for initialization of return values. The ASTs now
capture all of the temporaries we need to create, but
intentionally do not bind the tempoary that is actually returned,
so that it won't get destroyed twice.
- Make sure to perform an (elidable!) copy of the class object that
is returned from a class.
- Fix copy elision in CodeGen to properly see through the
subexpressions that occur with elidable copies.
- Give "new" its own entity kind; as with return values and thrown
objects, we don't bind the expression so we don't call a
destructor for it.
Note that, with this patch, I've broken returning move-only types in
C++0x. We'll fix it later, when we tackle NRVO.
llvm-svn: 91669
2009-12-18 13:02:21 +08:00
|
|
|
case SK_ZeroInitialization:
|
|
|
|
break;
|
2009-12-10 07:02:17 +08:00
|
|
|
}
|
2011-01-27 15:10:08 +08:00
|
|
|
|
|
|
|
// Walk through the computed steps for the initialization sequence,
|
2009-12-10 07:02:17 +08:00
|
|
|
// performing the specified conversions along the way.
|
2009-12-17 02:50:27 +08:00
|
|
|
bool ConstructorInitRequiresZeroInit = false;
|
2009-12-10 07:02:17 +08:00
|
|
|
for (step_iterator Step = step_begin(), StepEnd = step_end();
|
|
|
|
Step != StepEnd; ++Step) {
|
|
|
|
if (CurInit.isInvalid())
|
2010-08-27 07:41:50 +08:00
|
|
|
return ExprError();
|
2011-01-27 15:10:08 +08:00
|
|
|
|
2011-04-09 02:41:53 +08:00
|
|
|
QualType SourceType = CurInit.get() ? CurInit.get()->getType() : QualType();
|
2011-01-27 15:10:08 +08:00
|
|
|
|
2009-12-10 07:02:17 +08:00
|
|
|
switch (Step->Kind) {
|
|
|
|
case SK_ResolveAddressOfOverloadedFunction:
|
2011-01-27 15:10:08 +08:00
|
|
|
// Overload resolution determined which function invoke; update the
|
2009-12-10 07:02:17 +08:00
|
|
|
// initializer to reflect that choice.
|
2011-04-09 02:41:53 +08:00
|
|
|
S.CheckAddressOfMemberAccess(CurInit.get(), Step->Function.FoundDecl);
|
2013-05-04 14:44:46 +08:00
|
|
|
if (S.DiagnoseUseOfDecl(Step->Function.FoundDecl, Kind.getLocation()))
|
|
|
|
return ExprError();
|
2012-08-24 05:35:17 +08:00
|
|
|
CurInit = S.FixOverloadedFunctionReference(CurInit,
|
2010-03-31 05:47:33 +08:00
|
|
|
Step->Function.FoundDecl,
|
2010-03-19 15:35:19 +08:00
|
|
|
Step->Function.Function);
|
2009-12-10 07:02:17 +08:00
|
|
|
break;
|
2011-01-27 15:10:08 +08:00
|
|
|
|
2009-12-10 07:02:17 +08:00
|
|
|
case SK_CastDerivedToBaseRValue:
|
2010-07-20 12:20:21 +08:00
|
|
|
case SK_CastDerivedToBaseXValue:
|
2009-12-10 07:02:17 +08:00
|
|
|
case SK_CastDerivedToBaseLValue: {
|
|
|
|
// We have a derived-to-base cast that produces either an rvalue or an
|
|
|
|
// lvalue. Perform that cast.
|
2011-01-27 15:10:08 +08:00
|
|
|
|
2010-08-07 14:22:56 +08:00
|
|
|
CXXCastPath BasePath;
|
2010-04-25 03:06:50 +08:00
|
|
|
|
2009-12-10 07:02:17 +08:00
|
|
|
// Casts to inaccessible base classes are allowed with C-style casts.
|
|
|
|
bool IgnoreBaseAccess = Kind.isCStyleOrFunctionalCast();
|
|
|
|
if (S.CheckDerivedToBaseConversion(SourceType, Step->Type,
|
2011-04-09 02:41:53 +08:00
|
|
|
CurInit.get()->getLocStart(),
|
|
|
|
CurInit.get()->getSourceRange(),
|
2010-04-25 03:06:50 +08:00
|
|
|
&BasePath, IgnoreBaseAccess))
|
2010-08-27 07:41:50 +08:00
|
|
|
return ExprError();
|
2011-01-27 15:10:08 +08:00
|
|
|
|
Rework when and how vtables are emitted, by tracking where vtables are
"used" (e.g., we will refer to the vtable in the generated code) and
when they are defined (i.e., because we've seen the key function
definition). Previously, we were effectively tracking "potential
definitions" rather than uses, so we were a bit too eager about emitting
vtables for classes without key functions.
The new scheme:
- For every use of a vtable, Sema calls MarkVTableUsed() to indicate
the use. For example, this occurs when calling a virtual member
function of the class, defining a constructor of that class type,
dynamic_cast'ing from that type to a derived class, casting
to/through a virtual base class, etc.
- For every definition of a vtable, Sema calls MarkVTableUsed() to
indicate the definition. This happens at the end of the translation
unit for classes whose key function has been defined (so we can
delay computation of the key function; see PR6564), and will also
occur with explicit template instantiation definitions.
- For every vtable defined/used, we mark all of the virtual member
functions of that vtable as defined/used, unless we know that the key
function is in another translation unit. This instantiates virtual
member functions when needed.
- At the end of the translation unit, Sema tells CodeGen (via the
ASTConsumer) which vtables must be defined (CodeGen will define
them) and which may be used (for which CodeGen will define the
vtables lazily).
From a language perspective, both the old and the new schemes are
permissible: we're allowed to instantiate virtual member functions
whenever we want per the standard. However, all other C++ compilers
were more lazy than we were, and our eagerness was both a performance
issue (we instantiated too much) and a portability problem (we broke
Boost test cases, which now pass).
Notes:
(1) There's a ton of churn in the tests, because the order in which
vtables get emitted to IR has changed. I've tried to isolate some of
the larger tests from these issues.
(2) Some diagnostics related to
implicitly-instantiated/implicitly-defined virtual member functions
have moved to the point of first use/definition. It's better this
way.
(3) I could use a review of the places where we MarkVTableUsed, to
see if I missed any place where the language effectively requires a
vtable.
Fixes PR7114 and PR6564.
llvm-svn: 103718
2010-05-14 00:44:06 +08:00
|
|
|
if (S.BasePathInvolvesVirtualBase(BasePath)) {
|
|
|
|
QualType T = SourceType;
|
|
|
|
if (const PointerType *Pointer = T->getAs<PointerType>())
|
|
|
|
T = Pointer->getPointeeType();
|
|
|
|
if (const RecordType *RecordTy = T->getAs<RecordType>())
|
2011-04-09 02:41:53 +08:00
|
|
|
S.MarkVTableUsed(CurInit.get()->getLocStart(),
|
Rework when and how vtables are emitted, by tracking where vtables are
"used" (e.g., we will refer to the vtable in the generated code) and
when they are defined (i.e., because we've seen the key function
definition). Previously, we were effectively tracking "potential
definitions" rather than uses, so we were a bit too eager about emitting
vtables for classes without key functions.
The new scheme:
- For every use of a vtable, Sema calls MarkVTableUsed() to indicate
the use. For example, this occurs when calling a virtual member
function of the class, defining a constructor of that class type,
dynamic_cast'ing from that type to a derived class, casting
to/through a virtual base class, etc.
- For every definition of a vtable, Sema calls MarkVTableUsed() to
indicate the definition. This happens at the end of the translation
unit for classes whose key function has been defined (so we can
delay computation of the key function; see PR6564), and will also
occur with explicit template instantiation definitions.
- For every vtable defined/used, we mark all of the virtual member
functions of that vtable as defined/used, unless we know that the key
function is in another translation unit. This instantiates virtual
member functions when needed.
- At the end of the translation unit, Sema tells CodeGen (via the
ASTConsumer) which vtables must be defined (CodeGen will define
them) and which may be used (for which CodeGen will define the
vtables lazily).
From a language perspective, both the old and the new schemes are
permissible: we're allowed to instantiate virtual member functions
whenever we want per the standard. However, all other C++ compilers
were more lazy than we were, and our eagerness was both a performance
issue (we instantiated too much) and a portability problem (we broke
Boost test cases, which now pass).
Notes:
(1) There's a ton of churn in the tests, because the order in which
vtables get emitted to IR has changed. I've tried to isolate some of
the larger tests from these issues.
(2) Some diagnostics related to
implicitly-instantiated/implicitly-defined virtual member functions
have moved to the point of first use/definition. It's better this
way.
(3) I could use a review of the places where we MarkVTableUsed, to
see if I missed any place where the language effectively requires a
vtable.
Fixes PR7114 and PR6564.
llvm-svn: 103718
2010-05-14 00:44:06 +08:00
|
|
|
cast<CXXRecordDecl>(RecordTy->getDecl()));
|
|
|
|
}
|
|
|
|
|
2010-08-25 18:28:54 +08:00
|
|
|
ExprValueKind VK =
|
2010-07-20 12:20:21 +08:00
|
|
|
Step->Kind == SK_CastDerivedToBaseLValue ?
|
2010-08-25 18:28:54 +08:00
|
|
|
VK_LValue :
|
2010-07-20 12:20:21 +08:00
|
|
|
(Step->Kind == SK_CastDerivedToBaseXValue ?
|
2010-08-25 18:28:54 +08:00
|
|
|
VK_XValue :
|
|
|
|
VK_RValue);
|
2010-08-07 14:22:56 +08:00
|
|
|
CurInit = S.Owned(ImplicitCastExpr::Create(S.Context,
|
|
|
|
Step->Type,
|
2010-08-25 19:45:40 +08:00
|
|
|
CK_DerivedToBase,
|
2010-08-25 18:28:54 +08:00
|
|
|
CurInit.get(),
|
|
|
|
&BasePath, VK));
|
2009-12-10 07:02:17 +08:00
|
|
|
break;
|
|
|
|
}
|
2011-01-27 15:10:08 +08:00
|
|
|
|
2009-12-10 07:02:17 +08:00
|
|
|
case SK_BindReference:
|
2013-05-07 05:39:12 +08:00
|
|
|
// References cannot bind to bit-fields (C++ [dcl.init.ref]p5).
|
|
|
|
if (CurInit.get()->refersToBitField()) {
|
|
|
|
// We don't necessarily have an unambiguous source bit-field.
|
|
|
|
FieldDecl *BitField = CurInit.get()->getSourceBitField();
|
2009-12-10 07:02:17 +08:00
|
|
|
S.Diag(Kind.getLocation(), diag::err_reference_bind_to_bitfield)
|
2009-12-22 23:35:07 +08:00
|
|
|
<< Entity.getType().isVolatileQualified()
|
2013-05-07 05:39:12 +08:00
|
|
|
<< (BitField ? BitField->getDeclName() : DeclarationName())
|
|
|
|
<< (BitField != NULL)
|
2011-04-09 02:41:53 +08:00
|
|
|
<< CurInit.get()->getSourceRange();
|
2013-05-07 05:39:12 +08:00
|
|
|
if (BitField)
|
|
|
|
S.Diag(BitField->getLocation(), diag::note_bitfield_decl);
|
|
|
|
|
2010-08-27 07:41:50 +08:00
|
|
|
return ExprError();
|
2009-12-10 07:02:17 +08:00
|
|
|
}
|
2010-01-29 10:47:33 +08:00
|
|
|
|
2011-04-09 02:41:53 +08:00
|
|
|
if (CurInit.get()->refersToVectorElement()) {
|
2010-02-03 03:02:38 +08:00
|
|
|
// References cannot bind to vector elements.
|
2010-02-01 01:18:49 +08:00
|
|
|
S.Diag(Kind.getLocation(), diag::err_reference_bind_to_vector_element)
|
|
|
|
<< Entity.getType().isVolatileQualified()
|
2011-04-09 02:41:53 +08:00
|
|
|
<< CurInit.get()->getSourceRange();
|
2010-04-22 08:20:18 +08:00
|
|
|
PrintInitLocationNote(S, Entity);
|
2010-08-27 07:41:50 +08:00
|
|
|
return ExprError();
|
2010-02-01 01:18:49 +08:00
|
|
|
}
|
2011-01-27 15:10:08 +08:00
|
|
|
|
2009-12-10 07:02:17 +08:00
|
|
|
// Reference binding does not have any corresponding ASTs.
|
|
|
|
|
|
|
|
// Check exception specifications
|
2011-04-09 02:41:53 +08:00
|
|
|
if (S.CheckExceptionSpecCompatibility(CurInit.get(), DestType))
|
2010-08-27 07:41:50 +08:00
|
|
|
return ExprError();
|
2010-02-01 02:34:51 +08:00
|
|
|
|
2013-06-15 08:30:29 +08:00
|
|
|
// Even though we didn't materialize a temporary, the binding may still
|
|
|
|
// extend the lifetime of a temporary. This happens if we bind a reference
|
|
|
|
// to the result of a cast to reference type.
|
|
|
|
if (const ValueDecl *ExtendingDecl =
|
|
|
|
getDeclForTemporaryLifetimeExtension(Entity)) {
|
|
|
|
if (performReferenceExtension(CurInit.get(), ExtendingDecl))
|
|
|
|
warnOnLifetimeExtension(S, Entity, CurInit.get(), false,
|
|
|
|
ExtendingDecl);
|
|
|
|
}
|
|
|
|
|
2009-12-10 07:02:17 +08:00
|
|
|
break;
|
2010-02-01 02:34:51 +08:00
|
|
|
|
2013-06-05 08:46:14 +08:00
|
|
|
case SK_BindReferenceToTemporary: {
|
2013-04-11 08:58:58 +08:00
|
|
|
// Make sure the "temporary" is actually an rvalue.
|
|
|
|
assert(CurInit.get()->isRValue() && "not a temporary");
|
|
|
|
|
2009-12-10 07:02:17 +08:00
|
|
|
// Check exception specifications
|
2011-04-09 02:41:53 +08:00
|
|
|
if (S.CheckExceptionSpecCompatibility(CurInit.get(), DestType))
|
2010-08-27 07:41:50 +08:00
|
|
|
return ExprError();
|
2009-12-10 07:02:17 +08:00
|
|
|
|
2013-06-05 08:46:14 +08:00
|
|
|
// Maybe lifetime-extend the temporary's subobjects to match the
|
|
|
|
// entity's lifetime.
|
|
|
|
const ValueDecl *ExtendingDecl =
|
|
|
|
getDeclForTemporaryLifetimeExtension(Entity);
|
2013-06-13 05:51:50 +08:00
|
|
|
if (ExtendingDecl) {
|
2013-06-05 08:46:14 +08:00
|
|
|
performLifetimeExtension(CurInit.get(), ExtendingDecl);
|
2013-06-13 06:31:48 +08:00
|
|
|
warnOnLifetimeExtension(S, Entity, CurInit.get(), false, ExtendingDecl);
|
2013-06-13 05:51:50 +08:00
|
|
|
}
|
|
|
|
|
2011-06-22 01:03:29 +08:00
|
|
|
// Materialize the temporary into memory.
|
2013-06-13 04:42:33 +08:00
|
|
|
MaterializeTemporaryExpr *MTE = new (S.Context) MaterializeTemporaryExpr(
|
2013-06-05 08:46:14 +08:00
|
|
|
Entity.getType().getNonReferenceType(), CurInit.get(),
|
|
|
|
Entity.getType()->isLValueReferenceType(), ExtendingDecl);
|
2011-06-23 00:12:01 +08:00
|
|
|
|
|
|
|
// If we're binding to an Objective-C object that has lifetime, we
|
2013-06-13 04:42:33 +08:00
|
|
|
// need cleanups. Likewise if we're extending this temporary to automatic
|
|
|
|
// storage duration -- we need to register its cleanup during the
|
|
|
|
// full-expression's cleanups.
|
|
|
|
if ((S.getLangOpts().ObjCAutoRefCount &&
|
|
|
|
MTE->getType()->isObjCLifetimeType()) ||
|
|
|
|
(MTE->getStorageDuration() == SD_Automatic &&
|
|
|
|
MTE->getType().isDestructedType()))
|
2011-06-23 00:12:01 +08:00
|
|
|
S.ExprNeedsCleanups = true;
|
2013-06-13 04:42:33 +08:00
|
|
|
|
|
|
|
CurInit = S.Owned(MTE);
|
2009-12-10 07:02:17 +08:00
|
|
|
break;
|
2013-06-05 08:46:14 +08:00
|
|
|
}
|
2011-01-27 15:10:08 +08:00
|
|
|
|
2010-04-18 15:40:54 +08:00
|
|
|
case SK_ExtraneousCopyToTemporary:
|
2012-08-24 05:35:17 +08:00
|
|
|
CurInit = CopyObject(S, Step->Type, Entity, CurInit,
|
2010-04-18 15:40:54 +08:00
|
|
|
/*IsExtraneousCopy=*/true);
|
|
|
|
break;
|
|
|
|
|
2009-12-10 07:02:17 +08:00
|
|
|
case SK_UserConversion: {
|
|
|
|
// We have a user-defined conversion that invokes either a constructor
|
|
|
|
// or a conversion function.
|
2010-11-15 17:13:47 +08:00
|
|
|
CastKind CastKind;
|
Switch the initialization required by return statements over to the
new InitializationSequence. This fixes some bugs (e.g., PR5808),
changed some diagnostics, and caused more churn than expected. What's
new:
- InitializationSequence now has a "C conversion sequence" category
and step kind, which falls back to
- Changed the diagnostics for returns to always have the result type
of the function first and the type of the expression second.
CheckSingleAssignmentConstraints to peform checking in C.
- Improved ASTs for initialization of return values. The ASTs now
capture all of the temporaries we need to create, but
intentionally do not bind the tempoary that is actually returned,
so that it won't get destroyed twice.
- Make sure to perform an (elidable!) copy of the class object that
is returned from a class.
- Fix copy elision in CodeGen to properly see through the
subexpressions that occur with elidable copies.
- Give "new" its own entity kind; as with return values and thrown
objects, we don't bind the expression so we don't call a
destructor for it.
Note that, with this patch, I've broken returning move-only types in
C++0x. We'll fix it later, when we tackle NRVO.
llvm-svn: 91669
2009-12-18 13:02:21 +08:00
|
|
|
bool IsCopy = false;
|
2010-03-19 15:35:19 +08:00
|
|
|
FunctionDecl *Fn = Step->Function.Function;
|
|
|
|
DeclAccessPair FoundFn = Step->Function.FoundDecl;
|
2011-10-05 15:56:41 +08:00
|
|
|
bool HadMultipleCandidates = Step->Function.HadMultipleCandidates;
|
2010-04-25 07:45:46 +08:00
|
|
|
bool CreatedObject = false;
|
2010-02-01 11:16:54 +08:00
|
|
|
if (CXXConstructorDecl *Constructor = dyn_cast<CXXConstructorDecl>(Fn)) {
|
2009-12-10 07:02:17 +08:00
|
|
|
// Build a call to the selected constructor.
|
2012-08-24 06:51:59 +08:00
|
|
|
SmallVector<Expr*, 8> ConstructorArgs;
|
2011-04-09 02:41:53 +08:00
|
|
|
SourceLocation Loc = CurInit.get()->getLocStart();
|
2009-12-10 07:02:17 +08:00
|
|
|
CurInit.release(); // Ownership transferred into MultiExprArg, below.
|
2010-02-01 11:16:54 +08:00
|
|
|
|
2009-12-10 07:02:17 +08:00
|
|
|
// Determine the arguments required to actually perform the constructor
|
|
|
|
// call.
|
2011-04-09 02:41:53 +08:00
|
|
|
Expr *Arg = CurInit.get();
|
2009-12-10 07:02:17 +08:00
|
|
|
if (S.CompleteConstructorCall(Constructor,
|
2011-04-09 02:41:53 +08:00
|
|
|
MultiExprArg(&Arg, 1),
|
2009-12-10 07:02:17 +08:00
|
|
|
Loc, ConstructorArgs))
|
2010-08-27 07:41:50 +08:00
|
|
|
return ExprError();
|
2011-01-27 15:10:08 +08:00
|
|
|
|
2012-02-12 03:22:50 +08:00
|
|
|
// Build an expression that constructs a temporary.
|
2011-01-27 15:10:08 +08:00
|
|
|
CurInit = S.BuildCXXConstructExpr(Loc, Step->Type, Constructor,
|
2012-08-24 05:35:17 +08:00
|
|
|
ConstructorArgs,
|
2011-10-05 15:56:41 +08:00
|
|
|
HadMultipleCandidates,
|
2012-12-19 09:39:02 +08:00
|
|
|
/*ListInit*/ false,
|
2010-08-24 15:32:53 +08:00
|
|
|
/*ZeroInit*/ false,
|
2010-10-25 16:47:36 +08:00
|
|
|
CXXConstructExpr::CK_Complete,
|
|
|
|
SourceRange());
|
2009-12-10 07:02:17 +08:00
|
|
|
if (CurInit.isInvalid())
|
2010-08-27 07:41:50 +08:00
|
|
|
return ExprError();
|
2010-02-01 11:16:54 +08:00
|
|
|
|
2010-04-22 02:47:17 +08:00
|
|
|
S.CheckConstructorAccess(Kind.getLocation(), Constructor, Entity,
|
2010-03-19 15:35:19 +08:00
|
|
|
FoundFn.getAccess());
|
2013-05-04 14:44:46 +08:00
|
|
|
if (S.DiagnoseUseOfDecl(FoundFn, Kind.getLocation()))
|
|
|
|
return ExprError();
|
2011-01-27 15:10:08 +08:00
|
|
|
|
2010-08-25 19:45:40 +08:00
|
|
|
CastKind = CK_ConstructorConversion;
|
Switch the initialization required by return statements over to the
new InitializationSequence. This fixes some bugs (e.g., PR5808),
changed some diagnostics, and caused more churn than expected. What's
new:
- InitializationSequence now has a "C conversion sequence" category
and step kind, which falls back to
- Changed the diagnostics for returns to always have the result type
of the function first and the type of the expression second.
CheckSingleAssignmentConstraints to peform checking in C.
- Improved ASTs for initialization of return values. The ASTs now
capture all of the temporaries we need to create, but
intentionally do not bind the tempoary that is actually returned,
so that it won't get destroyed twice.
- Make sure to perform an (elidable!) copy of the class object that
is returned from a class.
- Fix copy elision in CodeGen to properly see through the
subexpressions that occur with elidable copies.
- Give "new" its own entity kind; as with return values and thrown
objects, we don't bind the expression so we don't call a
destructor for it.
Note that, with this patch, I've broken returning move-only types in
C++0x. We'll fix it later, when we tackle NRVO.
llvm-svn: 91669
2009-12-18 13:02:21 +08:00
|
|
|
QualType Class = S.Context.getTypeDeclType(Constructor->getParent());
|
|
|
|
if (S.Context.hasSameUnqualifiedType(SourceType, Class) ||
|
|
|
|
S.IsDerivedFrom(SourceType, Class))
|
|
|
|
IsCopy = true;
|
2011-01-27 15:10:08 +08:00
|
|
|
|
2010-04-25 07:45:46 +08:00
|
|
|
CreatedObject = true;
|
2009-12-10 07:02:17 +08:00
|
|
|
} else {
|
|
|
|
// Build a call to the conversion function.
|
2010-02-01 11:16:54 +08:00
|
|
|
CXXConversionDecl *Conversion = cast<CXXConversionDecl>(Fn);
|
2011-04-09 02:41:53 +08:00
|
|
|
S.CheckMemberOperatorAccess(Kind.getLocation(), CurInit.get(), 0,
|
2010-03-19 15:35:19 +08:00
|
|
|
FoundFn);
|
2013-05-04 14:44:46 +08:00
|
|
|
if (S.DiagnoseUseOfDecl(FoundFn, Kind.getLocation()))
|
|
|
|
return ExprError();
|
2011-01-27 15:10:08 +08:00
|
|
|
|
|
|
|
// FIXME: Should we move this initialization into a separate
|
2009-12-10 07:02:17 +08:00
|
|
|
// derived-to-base conversion? I believe the answer is "no", because
|
|
|
|
// we don't want to turn off access control here for c-style casts.
|
2011-04-09 02:41:53 +08:00
|
|
|
ExprResult CurInitExprRes =
|
|
|
|
S.PerformObjectArgumentInitialization(CurInit.take(), /*Qualifier=*/0,
|
|
|
|
FoundFn, Conversion);
|
|
|
|
if(CurInitExprRes.isInvalid())
|
2010-08-27 07:41:50 +08:00
|
|
|
return ExprError();
|
2012-08-24 05:35:17 +08:00
|
|
|
CurInit = CurInitExprRes;
|
2011-01-27 15:10:08 +08:00
|
|
|
|
2009-12-10 07:02:17 +08:00
|
|
|
// Build the actual call to the conversion function.
|
2011-10-05 15:56:41 +08:00
|
|
|
CurInit = S.BuildCXXMemberCallExpr(CurInit.get(), FoundFn, Conversion,
|
|
|
|
HadMultipleCandidates);
|
2009-12-10 07:02:17 +08:00
|
|
|
if (CurInit.isInvalid() || !CurInit.get())
|
2010-08-27 07:41:50 +08:00
|
|
|
return ExprError();
|
2011-01-27 15:10:08 +08:00
|
|
|
|
2010-08-25 19:45:40 +08:00
|
|
|
CastKind = CK_UserDefinedConversion;
|
2011-01-27 15:10:08 +08:00
|
|
|
|
2010-04-25 07:45:46 +08:00
|
|
|
CreatedObject = Conversion->getResultType()->isRecordType();
|
2009-12-10 07:02:17 +08:00
|
|
|
}
|
2011-01-27 15:10:08 +08:00
|
|
|
|
2011-07-15 03:07:55 +08:00
|
|
|
bool RequiresCopy = !IsCopy && !isReferenceBinding(Steps.back());
|
2011-11-17 06:46:05 +08:00
|
|
|
bool MaybeBindToTemp = RequiresCopy || shouldBindAsTemporary(Entity);
|
|
|
|
|
|
|
|
if (!MaybeBindToTemp && CreatedObject && shouldDestroyTemporary(Entity)) {
|
2011-04-09 02:41:53 +08:00
|
|
|
QualType T = CurInit.get()->getType();
|
2010-04-25 07:45:46 +08:00
|
|
|
if (const RecordType *Record = T->getAs<RecordType>()) {
|
2011-01-27 15:10:08 +08:00
|
|
|
CXXDestructorDecl *Destructor
|
2010-07-02 06:47:18 +08:00
|
|
|
= S.LookupDestructor(cast<CXXRecordDecl>(Record->getDecl()));
|
2011-04-09 02:41:53 +08:00
|
|
|
S.CheckDestructorAccess(CurInit.get()->getLocStart(), Destructor,
|
2010-04-25 07:45:46 +08:00
|
|
|
S.PDiag(diag::err_access_dtor_temp) << T);
|
2012-02-02 11:46:19 +08:00
|
|
|
S.MarkFunctionReferenced(CurInit.get()->getLocStart(), Destructor);
|
2013-05-04 14:44:46 +08:00
|
|
|
if (S.DiagnoseUseOfDecl(Destructor, CurInit.get()->getLocStart()))
|
|
|
|
return ExprError();
|
2010-04-25 07:45:46 +08:00
|
|
|
}
|
|
|
|
}
|
2011-01-27 15:10:08 +08:00
|
|
|
|
2010-08-07 14:22:56 +08:00
|
|
|
CurInit = S.Owned(ImplicitCastExpr::Create(S.Context,
|
2011-04-09 02:41:53 +08:00
|
|
|
CurInit.get()->getType(),
|
|
|
|
CastKind, CurInit.get(), 0,
|
2011-09-27 09:11:35 +08:00
|
|
|
CurInit.get()->getValueKind()));
|
2011-11-17 06:46:05 +08:00
|
|
|
if (MaybeBindToTemp)
|
|
|
|
CurInit = S.MaybeBindToTemporary(CurInit.takeAs<Expr>());
|
Rework our handling of copy construction of temporaries, which was a
poor (and wrong) approximation of the actual rules governing when to
build a copy and when it can be elided.
The correct implementation is actually simpler than the
approximation. When we only enumerate constructors as part of
initialization (e.g., for direct initialization or when we're copying
from a class type or one of its derived classes), we don't create a
copy. When we enumerate all conversion functions, we do create a
copy. Before, we created some extra copies and missed some
others. The new test copy-initialization.cpp shows a case where we
missed creating a (required, non-elidable) copy as part of a
user-defined conversion, which resulted in a miscompile. This commit
also fixes PR6757, where the missing copy made us reject well-formed
code in the ternary operator.
This commit also cleans up our handling of copy elision in the case
where we create an extra copy of a temporary object, which became
necessary now that we produce the right copies. The code that seeks to
find the temporary object being copied has moved into
Expr::getTemporaryObject(); it used to have two different
not-quite-the-same implementations, one in Sema and one in CodeGen.
Note that we still do not attempt to perform the named return value
optimization, so we miss copy elisions for return values and throw
expressions.
llvm-svn: 100196
2010-04-03 02:24:57 +08:00
|
|
|
if (RequiresCopy)
|
2010-04-18 15:40:54 +08:00
|
|
|
CurInit = CopyObject(S, Entity.getType().getNonReferenceType(), Entity,
|
2012-08-24 05:35:17 +08:00
|
|
|
CurInit, /*IsExtraneousCopy=*/false);
|
2009-12-10 07:02:17 +08:00
|
|
|
break;
|
|
|
|
}
|
2010-07-20 12:20:21 +08:00
|
|
|
|
2009-12-10 07:02:17 +08:00
|
|
|
case SK_QualificationConversionLValue:
|
2010-07-20 12:20:21 +08:00
|
|
|
case SK_QualificationConversionXValue:
|
|
|
|
case SK_QualificationConversionRValue: {
|
2009-12-10 07:02:17 +08:00
|
|
|
// Perform a qualification conversion; these can never go wrong.
|
2010-08-25 18:28:54 +08:00
|
|
|
ExprValueKind VK =
|
2010-07-20 12:20:21 +08:00
|
|
|
Step->Kind == SK_QualificationConversionLValue ?
|
2010-08-25 18:28:54 +08:00
|
|
|
VK_LValue :
|
2010-07-20 12:20:21 +08:00
|
|
|
(Step->Kind == SK_QualificationConversionXValue ?
|
2010-08-25 18:28:54 +08:00
|
|
|
VK_XValue :
|
|
|
|
VK_RValue);
|
2011-04-09 02:41:53 +08:00
|
|
|
CurInit = S.ImpCastExprToType(CurInit.take(), Step->Type, CK_NoOp, VK);
|
2009-12-10 07:02:17 +08:00
|
|
|
break;
|
2010-07-20 12:20:21 +08:00
|
|
|
}
|
|
|
|
|
2013-04-11 08:58:58 +08:00
|
|
|
case SK_LValueToRValue: {
|
|
|
|
assert(CurInit.get()->isGLValue() && "cannot load from a prvalue");
|
|
|
|
CurInit = S.Owned(ImplicitCastExpr::Create(S.Context, Step->Type,
|
|
|
|
CK_LValueToRValue,
|
|
|
|
CurInit.take(),
|
|
|
|
/*BasePath=*/0,
|
|
|
|
VK_RValue));
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2013-09-22 05:19:19 +08:00
|
|
|
case SK_ConversionSequence:
|
|
|
|
case SK_ConversionSequenceNoNarrowing: {
|
|
|
|
Sema::CheckedConversionKind CCK
|
2011-06-16 07:02:42 +08:00
|
|
|
= Kind.isCStyleCast()? Sema::CCK_CStyleCast
|
|
|
|
: Kind.isFunctionalCast()? Sema::CCK_FunctionalCast
|
2011-11-30 06:48:16 +08:00
|
|
|
: Kind.isExplicitCast()? Sema::CCK_OtherCast
|
2011-06-16 07:02:42 +08:00
|
|
|
: Sema::CCK_ImplicitConversion;
|
2011-04-09 02:41:53 +08:00
|
|
|
ExprResult CurInitExprRes =
|
|
|
|
S.PerformImplicitConversion(CurInit.get(), Step->Type, *Step->ICS,
|
2011-06-16 07:02:42 +08:00
|
|
|
getAssignmentAction(Entity), CCK);
|
2011-04-09 02:41:53 +08:00
|
|
|
if (CurInitExprRes.isInvalid())
|
2010-08-27 07:41:50 +08:00
|
|
|
return ExprError();
|
2012-08-24 05:35:17 +08:00
|
|
|
CurInit = CurInitExprRes;
|
2013-09-22 05:19:19 +08:00
|
|
|
|
|
|
|
if (Step->Kind == SK_ConversionSequenceNoNarrowing &&
|
|
|
|
S.getLangOpts().CPlusPlus && !CurInit.get()->isValueDependent())
|
|
|
|
DiagnoseNarrowingInInitList(S, *Step->ICS, SourceType, Entity.getType(),
|
|
|
|
CurInit.get());
|
2009-12-10 07:02:17 +08:00
|
|
|
break;
|
2010-04-17 03:30:02 +08:00
|
|
|
}
|
2011-01-27 15:10:08 +08:00
|
|
|
|
2009-12-11 01:56:55 +08:00
|
|
|
case SK_ListInitialization: {
|
2011-04-09 02:41:53 +08:00
|
|
|
InitListExpr *InitList = cast<InitListExpr>(CurInit.get());
|
2013-06-13 06:31:48 +08:00
|
|
|
// If we're not initializing the top-level entity, we need to create an
|
|
|
|
// InitializeTemporary entity for our target type.
|
|
|
|
QualType Ty = Step->Type;
|
|
|
|
bool IsTemporary = !S.Context.hasSameType(Entity.getType(), Ty);
|
2011-11-28 00:50:07 +08:00
|
|
|
InitializedEntity TempEntity = InitializedEntity::InitializeTemporary(Ty);
|
2013-02-02 09:13:06 +08:00
|
|
|
InitializedEntity InitEntity = IsTemporary ? TempEntity : Entity;
|
|
|
|
InitListChecker PerformInitList(S, InitEntity,
|
2013-06-06 19:41:05 +08:00
|
|
|
InitList, Ty, /*VerifyOnly=*/false);
|
2011-09-25 01:48:00 +08:00
|
|
|
if (PerformInitList.HadError())
|
2010-08-27 07:41:50 +08:00
|
|
|
return ExprError();
|
2009-12-11 01:56:55 +08:00
|
|
|
|
2013-06-13 06:31:48 +08:00
|
|
|
// Hack: We must update *ResultType if available in order to set the
|
|
|
|
// bounds of arrays, e.g. in 'int ar[] = {1, 2, 3};'.
|
|
|
|
// Worst case: 'const int (&arref)[] = {1, 2, 3};'.
|
|
|
|
if (ResultType &&
|
|
|
|
ResultType->getNonReferenceType()->isIncompleteArrayType()) {
|
2011-11-28 00:50:07 +08:00
|
|
|
if ((*ResultType)->isRValueReferenceType())
|
|
|
|
Ty = S.Context.getRValueReferenceType(Ty);
|
|
|
|
else if ((*ResultType)->isLValueReferenceType())
|
|
|
|
Ty = S.Context.getLValueReferenceType(Ty,
|
|
|
|
(*ResultType)->getAs<LValueReferenceType>()->isSpelledAsLValue());
|
|
|
|
*ResultType = Ty;
|
|
|
|
}
|
|
|
|
|
|
|
|
InitListExpr *StructuredInitList =
|
|
|
|
PerformInitList.getFullyStructuredList();
|
2009-12-11 01:56:55 +08:00
|
|
|
CurInit.release();
|
2013-02-02 09:13:06 +08:00
|
|
|
CurInit = shouldBindAsTemporary(InitEntity)
|
|
|
|
? S.MaybeBindToTemporary(StructuredInitList)
|
|
|
|
: S.Owned(StructuredInitList);
|
2009-12-11 01:56:55 +08:00
|
|
|
break;
|
|
|
|
}
|
2009-12-15 04:49:26 +08:00
|
|
|
|
2011-12-22 22:44:04 +08:00
|
|
|
case SK_ListConstructorCall: {
|
2012-02-13 00:37:24 +08:00
|
|
|
// When an initializer list is passed for a parameter of type "reference
|
|
|
|
// to object", we don't get an EK_Temporary entity, but instead an
|
|
|
|
// EK_Parameter entity with reference type.
|
2012-02-19 20:27:56 +08:00
|
|
|
// FIXME: This is a hack. What we really should do is create a user
|
|
|
|
// conversion step for this case, but this makes it considerably more
|
|
|
|
// complicated. For now, this will do.
|
2012-02-13 00:37:24 +08:00
|
|
|
InitializedEntity TempEntity = InitializedEntity::InitializeTemporary(
|
|
|
|
Entity.getType().getNonReferenceType());
|
|
|
|
bool UseTemporary = Entity.getType()->isReferenceType();
|
2012-07-05 16:39:21 +08:00
|
|
|
assert(Args.size() == 1 && "expected a single argument for list init");
|
2012-08-24 05:35:17 +08:00
|
|
|
InitListExpr *InitList = cast<InitListExpr>(Args[0]);
|
2012-04-19 14:58:00 +08:00
|
|
|
S.Diag(InitList->getExprLoc(), diag::warn_cxx98_compat_ctor_list_init)
|
|
|
|
<< InitList->getSourceRange();
|
2011-12-22 22:44:04 +08:00
|
|
|
MultiExprArg Arg(InitList->getInits(), InitList->getNumInits());
|
2012-02-13 00:37:24 +08:00
|
|
|
CurInit = PerformConstructorInitialization(S, UseTemporary ? TempEntity :
|
|
|
|
Entity,
|
2012-08-24 05:35:17 +08:00
|
|
|
Kind, Arg, *Step,
|
2012-12-19 09:39:02 +08:00
|
|
|
ConstructorInitRequiresZeroInit,
|
2013-09-07 13:49:53 +08:00
|
|
|
/*IsListInitialization*/ true,
|
|
|
|
InitList->getLBraceLoc(),
|
|
|
|
InitList->getRBraceLoc());
|
2011-12-22 22:44:04 +08:00
|
|
|
break;
|
|
|
|
}
|
2011-09-25 01:47:52 +08:00
|
|
|
|
2011-11-28 00:50:07 +08:00
|
|
|
case SK_UnwrapInitList:
|
|
|
|
CurInit = S.Owned(cast<InitListExpr>(CurInit.take())->getInit(0));
|
|
|
|
break;
|
|
|
|
|
|
|
|
case SK_RewrapInitList: {
|
|
|
|
Expr *E = CurInit.take();
|
|
|
|
InitListExpr *Syntactic = Step->WrappingSyntacticList;
|
|
|
|
InitListExpr *ILE = new (S.Context) InitListExpr(S.Context,
|
2012-08-24 19:54:20 +08:00
|
|
|
Syntactic->getLBraceLoc(), E, Syntactic->getRBraceLoc());
|
2011-11-28 00:50:07 +08:00
|
|
|
ILE->setSyntacticForm(Syntactic);
|
|
|
|
ILE->setType(E->getType());
|
|
|
|
ILE->setValueKind(E->getValueKind());
|
|
|
|
CurInit = S.Owned(ILE);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2012-02-19 20:27:56 +08:00
|
|
|
case SK_ConstructorInitialization: {
|
|
|
|
// When an initializer list is passed for a parameter of type "reference
|
|
|
|
// to object", we don't get an EK_Temporary entity, but instead an
|
|
|
|
// EK_Parameter entity with reference type.
|
|
|
|
// FIXME: This is a hack. What we really should do is create a user
|
|
|
|
// conversion step for this case, but this makes it considerably more
|
|
|
|
// complicated. For now, this will do.
|
|
|
|
InitializedEntity TempEntity = InitializedEntity::InitializeTemporary(
|
|
|
|
Entity.getType().getNonReferenceType());
|
|
|
|
bool UseTemporary = Entity.getType()->isReferenceType();
|
|
|
|
CurInit = PerformConstructorInitialization(S, UseTemporary ? TempEntity
|
|
|
|
: Entity,
|
2012-08-24 05:35:17 +08:00
|
|
|
Kind, Args, *Step,
|
2012-12-19 09:39:02 +08:00
|
|
|
ConstructorInitRequiresZeroInit,
|
2013-09-07 13:49:53 +08:00
|
|
|
/*IsListInitialization*/ false,
|
|
|
|
/*LBraceLoc*/ SourceLocation(),
|
|
|
|
/*RBraceLoc*/ SourceLocation());
|
2009-12-15 04:49:26 +08:00
|
|
|
break;
|
2012-02-19 20:27:56 +08:00
|
|
|
}
|
2011-01-27 15:10:08 +08:00
|
|
|
|
2009-12-15 08:01:57 +08:00
|
|
|
case SK_ZeroInitialization: {
|
2009-12-17 02:50:27 +08:00
|
|
|
step_iterator NextStep = Step;
|
|
|
|
++NextStep;
|
2011-01-27 15:10:08 +08:00
|
|
|
if (NextStep != StepEnd &&
|
2012-07-05 16:39:21 +08:00
|
|
|
(NextStep->Kind == SK_ConstructorInitialization ||
|
|
|
|
NextStep->Kind == SK_ListConstructorCall)) {
|
2009-12-17 02:50:27 +08:00
|
|
|
// The need for zero-initialization is recorded directly into
|
|
|
|
// the call to the object's constructor within the next step.
|
|
|
|
ConstructorInitRequiresZeroInit = true;
|
|
|
|
} else if (Kind.getKind() == InitializationKind::IK_Value &&
|
2012-03-11 15:00:24 +08:00
|
|
|
S.getLangOpts().CPlusPlus &&
|
2009-12-17 02:50:27 +08:00
|
|
|
!Kind.isImplicitValueInit()) {
|
2010-09-08 08:15:04 +08:00
|
|
|
TypeSourceInfo *TSInfo = Entity.getTypeSourceInfo();
|
|
|
|
if (!TSInfo)
|
2011-01-27 15:10:08 +08:00
|
|
|
TSInfo = S.Context.getTrivialTypeSourceInfo(Step->Type,
|
2010-09-08 08:15:04 +08:00
|
|
|
Kind.getRange().getBegin());
|
|
|
|
|
|
|
|
CurInit = S.Owned(new (S.Context) CXXScalarValueInitExpr(
|
|
|
|
TSInfo->getType().getNonLValueExprType(S.Context),
|
|
|
|
TSInfo,
|
2009-12-15 08:01:57 +08:00
|
|
|
Kind.getRange().getEnd()));
|
2009-12-17 02:50:27 +08:00
|
|
|
} else {
|
2009-12-15 08:01:57 +08:00
|
|
|
CurInit = S.Owned(new (S.Context) ImplicitValueInitExpr(Step->Type));
|
2009-12-17 02:50:27 +08:00
|
|
|
}
|
2009-12-15 08:01:57 +08:00
|
|
|
break;
|
|
|
|
}
|
Switch the initialization required by return statements over to the
new InitializationSequence. This fixes some bugs (e.g., PR5808),
changed some diagnostics, and caused more churn than expected. What's
new:
- InitializationSequence now has a "C conversion sequence" category
and step kind, which falls back to
- Changed the diagnostics for returns to always have the result type
of the function first and the type of the expression second.
CheckSingleAssignmentConstraints to peform checking in C.
- Improved ASTs for initialization of return values. The ASTs now
capture all of the temporaries we need to create, but
intentionally do not bind the tempoary that is actually returned,
so that it won't get destroyed twice.
- Make sure to perform an (elidable!) copy of the class object that
is returned from a class.
- Fix copy elision in CodeGen to properly see through the
subexpressions that occur with elidable copies.
- Give "new" its own entity kind; as with return values and thrown
objects, we don't bind the expression so we don't call a
destructor for it.
Note that, with this patch, I've broken returning move-only types in
C++0x. We'll fix it later, when we tackle NRVO.
llvm-svn: 91669
2009-12-18 13:02:21 +08:00
|
|
|
|
|
|
|
case SK_CAssignment: {
|
2011-04-09 02:41:53 +08:00
|
|
|
QualType SourceType = CurInit.get()->getType();
|
2012-08-24 05:35:17 +08:00
|
|
|
ExprResult Result = CurInit;
|
Switch the initialization required by return statements over to the
new InitializationSequence. This fixes some bugs (e.g., PR5808),
changed some diagnostics, and caused more churn than expected. What's
new:
- InitializationSequence now has a "C conversion sequence" category
and step kind, which falls back to
- Changed the diagnostics for returns to always have the result type
of the function first and the type of the expression second.
CheckSingleAssignmentConstraints to peform checking in C.
- Improved ASTs for initialization of return values. The ASTs now
capture all of the temporaries we need to create, but
intentionally do not bind the tempoary that is actually returned,
so that it won't get destroyed twice.
- Make sure to perform an (elidable!) copy of the class object that
is returned from a class.
- Fix copy elision in CodeGen to properly see through the
subexpressions that occur with elidable copies.
- Give "new" its own entity kind; as with return values and thrown
objects, we don't bind the expression so we don't call a
destructor for it.
Note that, with this patch, I've broken returning move-only types in
C++0x. We'll fix it later, when we tackle NRVO.
llvm-svn: 91669
2009-12-18 13:02:21 +08:00
|
|
|
Sema::AssignConvertType ConvTy =
|
2013-08-01 05:40:51 +08:00
|
|
|
S.CheckSingleAssignmentConstraints(Step->Type, Result, true,
|
|
|
|
Entity.getKind() == InitializedEntity::EK_Parameter_CF_Audited);
|
2011-04-09 02:41:53 +08:00
|
|
|
if (Result.isInvalid())
|
|
|
|
return ExprError();
|
2012-08-24 05:35:17 +08:00
|
|
|
CurInit = Result;
|
2009-12-22 15:24:36 +08:00
|
|
|
|
|
|
|
// If this is a call, allow conversion to a transparent union.
|
2012-08-24 05:35:17 +08:00
|
|
|
ExprResult CurInitExprRes = CurInit;
|
2009-12-22 15:24:36 +08:00
|
|
|
if (ConvTy != Sema::Compatible &&
|
2013-08-01 02:21:45 +08:00
|
|
|
Entity.isParameterKind() &&
|
2011-04-09 02:41:53 +08:00
|
|
|
S.CheckTransparentUnionArgumentConstraints(Step->Type, CurInitExprRes)
|
2009-12-22 15:24:36 +08:00
|
|
|
== Sema::Compatible)
|
|
|
|
ConvTy = Sema::Compatible;
|
2011-04-09 02:41:53 +08:00
|
|
|
if (CurInitExprRes.isInvalid())
|
|
|
|
return ExprError();
|
2012-08-24 05:35:17 +08:00
|
|
|
CurInit = CurInitExprRes;
|
2009-12-22 15:24:36 +08:00
|
|
|
|
2010-04-22 08:20:18 +08:00
|
|
|
bool Complained;
|
Switch the initialization required by return statements over to the
new InitializationSequence. This fixes some bugs (e.g., PR5808),
changed some diagnostics, and caused more churn than expected. What's
new:
- InitializationSequence now has a "C conversion sequence" category
and step kind, which falls back to
- Changed the diagnostics for returns to always have the result type
of the function first and the type of the expression second.
CheckSingleAssignmentConstraints to peform checking in C.
- Improved ASTs for initialization of return values. The ASTs now
capture all of the temporaries we need to create, but
intentionally do not bind the tempoary that is actually returned,
so that it won't get destroyed twice.
- Make sure to perform an (elidable!) copy of the class object that
is returned from a class.
- Fix copy elision in CodeGen to properly see through the
subexpressions that occur with elidable copies.
- Give "new" its own entity kind; as with return values and thrown
objects, we don't bind the expression so we don't call a
destructor for it.
Note that, with this patch, I've broken returning move-only types in
C++0x. We'll fix it later, when we tackle NRVO.
llvm-svn: 91669
2009-12-18 13:02:21 +08:00
|
|
|
if (S.DiagnoseAssignmentResult(ConvTy, Kind.getLocation(),
|
|
|
|
Step->Type, SourceType,
|
2011-04-09 02:41:53 +08:00
|
|
|
CurInit.get(),
|
2013-08-01 07:19:34 +08:00
|
|
|
getAssignmentAction(Entity, true),
|
2010-04-22 08:20:18 +08:00
|
|
|
&Complained)) {
|
|
|
|
PrintInitLocationNote(S, Entity);
|
2010-08-27 07:41:50 +08:00
|
|
|
return ExprError();
|
2010-04-22 08:20:18 +08:00
|
|
|
} else if (Complained)
|
|
|
|
PrintInitLocationNote(S, Entity);
|
Switch the initialization required by return statements over to the
new InitializationSequence. This fixes some bugs (e.g., PR5808),
changed some diagnostics, and caused more churn than expected. What's
new:
- InitializationSequence now has a "C conversion sequence" category
and step kind, which falls back to
- Changed the diagnostics for returns to always have the result type
of the function first and the type of the expression second.
CheckSingleAssignmentConstraints to peform checking in C.
- Improved ASTs for initialization of return values. The ASTs now
capture all of the temporaries we need to create, but
intentionally do not bind the tempoary that is actually returned,
so that it won't get destroyed twice.
- Make sure to perform an (elidable!) copy of the class object that
is returned from a class.
- Fix copy elision in CodeGen to properly see through the
subexpressions that occur with elidable copies.
- Give "new" its own entity kind; as with return values and thrown
objects, we don't bind the expression so we don't call a
destructor for it.
Note that, with this patch, I've broken returning move-only types in
C++0x. We'll fix it later, when we tackle NRVO.
llvm-svn: 91669
2009-12-18 13:02:21 +08:00
|
|
|
break;
|
|
|
|
}
|
2009-12-19 16:11:05 +08:00
|
|
|
|
|
|
|
case SK_StringInit: {
|
|
|
|
QualType Ty = Step->Type;
|
2011-04-09 02:41:53 +08:00
|
|
|
CheckStringInit(CurInit.get(), ResultType ? *ResultType : Ty,
|
2011-02-21 15:57:55 +08:00
|
|
|
S.Context.getAsArrayType(Ty), S);
|
2009-12-19 16:11:05 +08:00
|
|
|
break;
|
|
|
|
}
|
2010-08-07 19:51:51 +08:00
|
|
|
|
|
|
|
case SK_ObjCObjectConversion:
|
2011-04-09 02:41:53 +08:00
|
|
|
CurInit = S.ImpCastExprToType(CurInit.take(), Step->Type,
|
2010-08-25 19:45:40 +08:00
|
|
|
CK_ObjCObjectLValueCast,
|
2011-09-28 05:58:52 +08:00
|
|
|
CurInit.get()->getValueKind());
|
2010-08-07 19:51:51 +08:00
|
|
|
break;
|
2011-02-23 02:29:51 +08:00
|
|
|
|
|
|
|
case SK_ArrayInit:
|
|
|
|
// Okay: we checked everything before creating this step. Note that
|
|
|
|
// this is a GNU extension.
|
|
|
|
S.Diag(Kind.getLocation(), diag::ext_array_init_copy)
|
2011-04-09 02:41:53 +08:00
|
|
|
<< Step->Type << CurInit.get()->getType()
|
|
|
|
<< CurInit.get()->getSourceRange();
|
2011-02-23 02:29:51 +08:00
|
|
|
|
|
|
|
// If the destination type is an incomplete array type, update the
|
|
|
|
// type accordingly.
|
|
|
|
if (ResultType) {
|
|
|
|
if (const IncompleteArrayType *IncompleteDest
|
|
|
|
= S.Context.getAsIncompleteArrayType(Step->Type)) {
|
|
|
|
if (const ConstantArrayType *ConstantSource
|
2011-04-09 02:41:53 +08:00
|
|
|
= S.Context.getAsConstantArrayType(CurInit.get()->getType())) {
|
2011-02-23 02:29:51 +08:00
|
|
|
*ResultType = S.Context.getConstantArrayType(
|
|
|
|
IncompleteDest->getElementType(),
|
|
|
|
ConstantSource->getSize(),
|
|
|
|
ArrayType::Normal, 0);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2011-06-16 07:02:42 +08:00
|
|
|
break;
|
2011-02-23 02:29:51 +08:00
|
|
|
|
2012-02-16 06:38:09 +08:00
|
|
|
case SK_ParenthesizedArrayInit:
|
|
|
|
// Okay: we checked everything before creating this step. Note that
|
|
|
|
// this is a GNU extension.
|
|
|
|
S.Diag(Kind.getLocation(), diag::ext_array_init_parens)
|
|
|
|
<< CurInit.get()->getSourceRange();
|
|
|
|
break;
|
|
|
|
|
2011-06-16 07:02:42 +08:00
|
|
|
case SK_PassByIndirectCopyRestore:
|
|
|
|
case SK_PassByIndirectRestore:
|
|
|
|
checkIndirectCopyRestoreSource(S, CurInit.get());
|
|
|
|
CurInit = S.Owned(new (S.Context)
|
|
|
|
ObjCIndirectCopyRestoreExpr(CurInit.take(), Step->Type,
|
|
|
|
Step->Kind == SK_PassByIndirectCopyRestore));
|
|
|
|
break;
|
|
|
|
|
|
|
|
case SK_ProduceObjCObject:
|
|
|
|
CurInit = S.Owned(ImplicitCastExpr::Create(S.Context, Step->Type,
|
2011-09-10 14:18:15 +08:00
|
|
|
CK_ARCProduceObject,
|
2011-06-16 07:02:42 +08:00
|
|
|
CurInit.take(), 0, VK_RValue));
|
2011-02-23 02:29:51 +08:00
|
|
|
break;
|
2012-01-18 06:49:42 +08:00
|
|
|
|
|
|
|
case SK_StdInitializerList: {
|
2013-06-13 06:31:48 +08:00
|
|
|
S.Diag(CurInit.get()->getExprLoc(),
|
|
|
|
diag::warn_cxx98_compat_initializer_list_init)
|
|
|
|
<< CurInit.get()->getSourceRange();
|
2012-03-06 03:35:43 +08:00
|
|
|
|
2013-06-13 06:31:48 +08:00
|
|
|
// Maybe lifetime-extend the array temporary's subobjects to match the
|
|
|
|
// entity's lifetime.
|
|
|
|
const ValueDecl *ExtendingDecl =
|
|
|
|
getDeclForTemporaryLifetimeExtension(Entity);
|
|
|
|
if (ExtendingDecl) {
|
|
|
|
performLifetimeExtension(CurInit.get(), ExtendingDecl);
|
|
|
|
warnOnLifetimeExtension(S, Entity, CurInit.get(), true, ExtendingDecl);
|
2012-01-18 06:49:42 +08:00
|
|
|
}
|
2013-06-13 06:31:48 +08:00
|
|
|
|
|
|
|
// Materialize the temporary into memory.
|
|
|
|
MaterializeTemporaryExpr *MTE = new (S.Context)
|
|
|
|
MaterializeTemporaryExpr(CurInit.get()->getType(), CurInit.get(),
|
|
|
|
/*lvalue reference*/ false, ExtendingDecl);
|
|
|
|
|
|
|
|
// Wrap it in a construction of a std::initializer_list<T>.
|
|
|
|
CurInit = S.Owned(
|
|
|
|
new (S.Context) CXXStdInitializerListExpr(Step->Type, MTE));
|
|
|
|
|
|
|
|
// Bind the result, in case the library has given initializer_list a
|
|
|
|
// non-trivial destructor.
|
|
|
|
if (shouldBindAsTemporary(Entity))
|
|
|
|
CurInit = S.MaybeBindToTemporary(CurInit.take());
|
2012-01-18 06:49:42 +08:00
|
|
|
break;
|
|
|
|
}
|
2013-06-13 06:31:48 +08:00
|
|
|
|
2013-02-07 18:55:47 +08:00
|
|
|
case SK_OCLSamplerInit: {
|
|
|
|
assert(Step->Type->isSamplerT() &&
|
2013-12-05 12:47:09 +08:00
|
|
|
"Sampler initialization on non-sampler type.");
|
2013-02-07 18:55:47 +08:00
|
|
|
|
|
|
|
QualType SourceType = CurInit.get()->getType();
|
|
|
|
|
2013-08-01 02:21:45 +08:00
|
|
|
if (Entity.isParameterKind()) {
|
2013-02-07 18:55:47 +08:00
|
|
|
if (!SourceType->isSamplerT())
|
|
|
|
S.Diag(Kind.getLocation(), diag::err_sampler_argument_required)
|
|
|
|
<< SourceType;
|
2013-08-01 02:21:45 +08:00
|
|
|
} else if (Entity.getKind() != InitializedEntity::EK_Variable) {
|
2013-02-07 18:55:47 +08:00
|
|
|
llvm_unreachable("Invalid EntityKind!");
|
|
|
|
}
|
|
|
|
|
|
|
|
break;
|
|
|
|
}
|
2013-01-20 20:31:11 +08:00
|
|
|
case SK_OCLZeroEvent: {
|
|
|
|
assert(Step->Type->isEventT() &&
|
2013-12-05 12:47:09 +08:00
|
|
|
"Event initialization on non-event type.");
|
2013-01-20 20:31:11 +08:00
|
|
|
|
|
|
|
CurInit = S.ImpCastExprToType(CurInit.take(), Step->Type,
|
|
|
|
CK_ZeroToOCLEvent,
|
|
|
|
CurInit.get()->getValueKind());
|
|
|
|
break;
|
|
|
|
}
|
2009-12-10 07:02:17 +08:00
|
|
|
}
|
|
|
|
}
|
2010-11-11 11:21:53 +08:00
|
|
|
|
|
|
|
// Diagnose non-fatal problems with the completed initialization.
|
|
|
|
if (Entity.getKind() == InitializedEntity::EK_Member &&
|
|
|
|
cast<FieldDecl>(Entity.getDecl())->isBitField())
|
|
|
|
S.CheckBitFieldInitialization(Kind.getLocation(),
|
|
|
|
cast<FieldDecl>(Entity.getDecl()),
|
|
|
|
CurInit.get());
|
2011-01-27 15:10:08 +08:00
|
|
|
|
2012-08-24 05:35:17 +08:00
|
|
|
return CurInit;
|
2009-12-10 07:02:17 +08:00
|
|
|
}
|
|
|
|
|
2012-12-08 10:01:17 +08:00
|
|
|
/// Somewhere within T there is an uninitialized reference subobject.
|
|
|
|
/// Dig it out and diagnose it.
|
2013-02-15 20:30:38 +08:00
|
|
|
static bool DiagnoseUninitializedReference(Sema &S, SourceLocation Loc,
|
|
|
|
QualType T) {
|
2012-12-08 10:01:17 +08:00
|
|
|
if (T->isReferenceType()) {
|
|
|
|
S.Diag(Loc, diag::err_reference_without_init)
|
|
|
|
<< T.getNonReferenceType();
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
CXXRecordDecl *RD = T->getBaseElementTypeUnsafe()->getAsCXXRecordDecl();
|
|
|
|
if (!RD || !RD->hasUninitializedReferenceMember())
|
|
|
|
return false;
|
|
|
|
|
|
|
|
for (CXXRecordDecl::field_iterator FI = RD->field_begin(),
|
|
|
|
FE = RD->field_end(); FI != FE; ++FI) {
|
|
|
|
if (FI->isUnnamedBitfield())
|
|
|
|
continue;
|
|
|
|
|
|
|
|
if (DiagnoseUninitializedReference(S, FI->getLocation(), FI->getType())) {
|
|
|
|
S.Diag(Loc, diag::note_value_initialization_here) << RD;
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
for (CXXRecordDecl::base_class_iterator BI = RD->bases_begin(),
|
|
|
|
BE = RD->bases_end();
|
|
|
|
BI != BE; ++BI) {
|
|
|
|
if (DiagnoseUninitializedReference(S, BI->getLocStart(), BI->getType())) {
|
|
|
|
S.Diag(Loc, diag::note_value_initialization_here) << RD;
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2009-12-10 07:02:17 +08:00
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
// Diagnose initialization failures
|
|
|
|
//===----------------------------------------------------------------------===//
|
2013-03-19 15:04:25 +08:00
|
|
|
|
|
|
|
/// Emit notes associated with an initialization that failed due to a
|
|
|
|
/// "simple" conversion failure.
|
|
|
|
static void emitBadConversionNotes(Sema &S, const InitializedEntity &entity,
|
|
|
|
Expr *op) {
|
|
|
|
QualType destType = entity.getType();
|
|
|
|
if (destType.getNonReferenceType()->isObjCObjectPointerType() &&
|
|
|
|
op->getType()->isObjCObjectPointerType()) {
|
|
|
|
|
|
|
|
// Emit a possible note about the conversion failing because the
|
|
|
|
// operand is a message send with a related result type.
|
|
|
|
S.EmitRelatedResultTypeNote(op);
|
|
|
|
|
|
|
|
// Emit a possible note about a return failing because we're
|
|
|
|
// expecting a related result type.
|
|
|
|
if (entity.getKind() == InitializedEntity::EK_Result)
|
|
|
|
S.EmitRelatedResultTypeNoteForReturn(destType);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-11-22 07:30:57 +08:00
|
|
|
static void diagnoseListInit(Sema &S, const InitializedEntity &Entity,
|
|
|
|
InitListExpr *InitList) {
|
|
|
|
QualType DestType = Entity.getType();
|
|
|
|
|
|
|
|
QualType E;
|
|
|
|
if (S.getLangOpts().CPlusPlus11 && S.isStdInitializerList(DestType, &E)) {
|
|
|
|
QualType ArrayType = S.Context.getConstantArrayType(
|
|
|
|
E.withConst(),
|
|
|
|
llvm::APInt(S.Context.getTypeSize(S.Context.getSizeType()),
|
|
|
|
InitList->getNumInits()),
|
|
|
|
clang::ArrayType::Normal, 0);
|
|
|
|
InitializedEntity HiddenArray =
|
|
|
|
InitializedEntity::InitializeTemporary(ArrayType);
|
|
|
|
return diagnoseListInit(S, HiddenArray, InitList);
|
|
|
|
}
|
|
|
|
|
|
|
|
InitListChecker DiagnoseInitList(S, Entity, InitList, DestType,
|
|
|
|
/*VerifyOnly=*/false);
|
|
|
|
assert(DiagnoseInitList.HadError() &&
|
|
|
|
"Inconsistent init list check result.");
|
|
|
|
}
|
|
|
|
|
2011-01-27 15:10:08 +08:00
|
|
|
bool InitializationSequence::Diagnose(Sema &S,
|
2009-12-10 07:02:17 +08:00
|
|
|
const InitializedEntity &Entity,
|
|
|
|
const InitializationKind &Kind,
|
2013-05-03 23:05:50 +08:00
|
|
|
ArrayRef<Expr *> Args) {
|
2011-06-05 21:59:05 +08:00
|
|
|
if (!Failed())
|
2009-12-10 07:02:17 +08:00
|
|
|
return false;
|
2011-01-27 15:10:08 +08:00
|
|
|
|
2009-12-22 23:35:07 +08:00
|
|
|
QualType DestType = Entity.getType();
|
2009-12-10 07:02:17 +08:00
|
|
|
switch (Failure) {
|
|
|
|
case FK_TooManyInitsForReference:
|
Rework base and member initialization in constructors, with several
(necessarily simultaneous) changes:
- CXXBaseOrMemberInitializer now contains only a single initializer
rather than a set of initialiation arguments + a constructor. The
single initializer covers all aspects of initialization, including
constructor calls as necessary but also cleanup of temporaries
created by the initializer (which we never handled
before!).
- Rework + simplify code generation for CXXBaseOrMemberInitializers,
since we can now just emit the initializer as an initializer.
- Switched base and member initialization over to the new
initialization code (InitializationSequence), so that it
- Improved diagnostics for the new initialization code when
initializing bases and members, to match the diagnostics produced
by the previous (special-purpose) code.
- Simplify the representation of type-checked constructor initializers in
templates; instead of keeping the fully-type-checked AST, which is
rather hard to undo at template instantiation time, throw away the
type-checked AST and store the raw expressions in the AST. This
simplifies instantiation, but loses a little but of information in
the AST.
- When type-checking implicit base or member initializers within a
dependent context, don't add the generated initializers into the
AST, because they'll look like they were explicit.
- Record in CXXConstructExpr when the constructor call is to
initialize a base class, so that CodeGen does not have to infer it
from context. This ensures that we call the right kind of
constructor.
There are also a few "opportunity" fixes here that were needed to not
regress, for example:
- Diagnose default-initialization of a const-qualified class that
does not have a user-declared default constructor. We had this
diagnostic specifically for bases and members, but missed it for
variables. That's fixed now.
- When defining the implicit constructors, destructor, and
copy-assignment operator, set the CurContext to that constructor
when we're defining the body.
llvm-svn: 94952
2010-01-31 17:12:51 +08:00
|
|
|
// FIXME: Customize for the initialized entity?
|
2013-05-03 23:05:50 +08:00
|
|
|
if (Args.empty()) {
|
2012-12-08 10:01:17 +08:00
|
|
|
// Dig out the reference subobject which is uninitialized and diagnose it.
|
|
|
|
// If this is value-initialization, this could be nested some way within
|
|
|
|
// the target type.
|
|
|
|
assert(Kind.getKind() == InitializationKind::IK_Value ||
|
|
|
|
DestType->isReferenceType());
|
|
|
|
bool Diagnosed =
|
|
|
|
DiagnoseUninitializedReference(S, Kind.getLocation(), DestType);
|
|
|
|
assert(Diagnosed && "couldn't find uninitialized reference to diagnose");
|
|
|
|
(void)Diagnosed;
|
|
|
|
} else // FIXME: diagnostic below could be better!
|
Rework base and member initialization in constructors, with several
(necessarily simultaneous) changes:
- CXXBaseOrMemberInitializer now contains only a single initializer
rather than a set of initialiation arguments + a constructor. The
single initializer covers all aspects of initialization, including
constructor calls as necessary but also cleanup of temporaries
created by the initializer (which we never handled
before!).
- Rework + simplify code generation for CXXBaseOrMemberInitializers,
since we can now just emit the initializer as an initializer.
- Switched base and member initialization over to the new
initialization code (InitializationSequence), so that it
- Improved diagnostics for the new initialization code when
initializing bases and members, to match the diagnostics produced
by the previous (special-purpose) code.
- Simplify the representation of type-checked constructor initializers in
templates; instead of keeping the fully-type-checked AST, which is
rather hard to undo at template instantiation time, throw away the
type-checked AST and store the raw expressions in the AST. This
simplifies instantiation, but loses a little but of information in
the AST.
- When type-checking implicit base or member initializers within a
dependent context, don't add the generated initializers into the
AST, because they'll look like they were explicit.
- Record in CXXConstructExpr when the constructor call is to
initialize a base class, so that CodeGen does not have to infer it
from context. This ensures that we call the right kind of
constructor.
There are also a few "opportunity" fixes here that were needed to not
regress, for example:
- Diagnose default-initialization of a const-qualified class that
does not have a user-declared default constructor. We had this
diagnostic specifically for bases and members, but missed it for
variables. That's fixed now.
- When defining the implicit constructors, destructor, and
copy-assignment operator, set the CurContext to that constructor
when we're defining the body.
llvm-svn: 94952
2010-01-31 17:12:51 +08:00
|
|
|
S.Diag(Kind.getLocation(), diag::err_reference_has_multiple_inits)
|
2013-05-03 23:05:50 +08:00
|
|
|
<< SourceRange(Args.front()->getLocStart(), Args.back()->getLocEnd());
|
2009-12-10 07:02:17 +08:00
|
|
|
break;
|
2011-01-27 15:10:08 +08:00
|
|
|
|
2009-12-10 07:02:17 +08:00
|
|
|
case FK_ArrayNeedsInitList:
|
2013-05-15 19:03:04 +08:00
|
|
|
S.Diag(Kind.getLocation(), diag::err_array_init_not_init_list) << 0;
|
|
|
|
break;
|
2009-12-10 07:02:17 +08:00
|
|
|
case FK_ArrayNeedsInitListOrStringLiteral:
|
2013-05-15 19:03:04 +08:00
|
|
|
S.Diag(Kind.getLocation(), diag::err_array_init_not_init_list) << 1;
|
|
|
|
break;
|
|
|
|
case FK_ArrayNeedsInitListOrWideStringLiteral:
|
|
|
|
S.Diag(Kind.getLocation(), diag::err_array_init_not_init_list) << 2;
|
|
|
|
break;
|
|
|
|
case FK_NarrowStringIntoWideCharArray:
|
|
|
|
S.Diag(Kind.getLocation(), diag::err_array_init_narrow_string_into_wchar);
|
|
|
|
break;
|
|
|
|
case FK_WideStringIntoCharArray:
|
|
|
|
S.Diag(Kind.getLocation(), diag::err_array_init_wide_string_into_char);
|
|
|
|
break;
|
|
|
|
case FK_IncompatWideStringIntoWideChar:
|
|
|
|
S.Diag(Kind.getLocation(),
|
|
|
|
diag::err_array_init_incompat_wide_string_into_wchar);
|
2009-12-10 07:02:17 +08:00
|
|
|
break;
|
2011-02-23 02:29:51 +08:00
|
|
|
case FK_ArrayTypeMismatch:
|
|
|
|
case FK_NonConstantArrayInit:
|
2013-11-22 07:30:57 +08:00
|
|
|
S.Diag(Kind.getLocation(),
|
2011-02-23 02:29:51 +08:00
|
|
|
(Failure == FK_ArrayTypeMismatch
|
|
|
|
? diag::err_array_init_different_type
|
|
|
|
: diag::err_array_init_non_constant_array))
|
|
|
|
<< DestType.getNonReferenceType()
|
|
|
|
<< Args[0]->getType()
|
|
|
|
<< Args[0]->getSourceRange();
|
|
|
|
break;
|
|
|
|
|
2012-01-05 08:13:19 +08:00
|
|
|
case FK_VariableLengthArrayHasInitializer:
|
|
|
|
S.Diag(Kind.getLocation(), diag::err_variable_object_no_init)
|
|
|
|
<< Args[0]->getSourceRange();
|
|
|
|
break;
|
|
|
|
|
2010-03-31 05:47:33 +08:00
|
|
|
case FK_AddressOfOverloadFailed: {
|
|
|
|
DeclAccessPair Found;
|
2011-01-27 15:10:08 +08:00
|
|
|
S.ResolveAddressOfOverloadedFunction(Args[0],
|
2009-12-10 07:02:17 +08:00
|
|
|
DestType.getNonReferenceType(),
|
2010-03-31 05:47:33 +08:00
|
|
|
true,
|
|
|
|
Found);
|
2009-12-10 07:02:17 +08:00
|
|
|
break;
|
2010-03-31 05:47:33 +08:00
|
|
|
}
|
2011-01-27 15:10:08 +08:00
|
|
|
|
2009-12-10 07:02:17 +08:00
|
|
|
case FK_ReferenceInitOverloadFailed:
|
2009-12-15 01:27:33 +08:00
|
|
|
case FK_UserConversionOverloadFailed:
|
2009-12-10 07:02:17 +08:00
|
|
|
switch (FailedOverloadResult) {
|
|
|
|
case OR_Ambiguous:
|
Switch the initialization required by return statements over to the
new InitializationSequence. This fixes some bugs (e.g., PR5808),
changed some diagnostics, and caused more churn than expected. What's
new:
- InitializationSequence now has a "C conversion sequence" category
and step kind, which falls back to
- Changed the diagnostics for returns to always have the result type
of the function first and the type of the expression second.
CheckSingleAssignmentConstraints to peform checking in C.
- Improved ASTs for initialization of return values. The ASTs now
capture all of the temporaries we need to create, but
intentionally do not bind the tempoary that is actually returned,
so that it won't get destroyed twice.
- Make sure to perform an (elidable!) copy of the class object that
is returned from a class.
- Fix copy elision in CodeGen to properly see through the
subexpressions that occur with elidable copies.
- Give "new" its own entity kind; as with return values and thrown
objects, we don't bind the expression so we don't call a
destructor for it.
Note that, with this patch, I've broken returning move-only types in
C++0x. We'll fix it later, when we tackle NRVO.
llvm-svn: 91669
2009-12-18 13:02:21 +08:00
|
|
|
if (Failure == FK_UserConversionOverloadFailed)
|
|
|
|
S.Diag(Kind.getLocation(), diag::err_typecheck_ambiguous_condition)
|
|
|
|
<< Args[0]->getType() << DestType
|
|
|
|
<< Args[0]->getSourceRange();
|
|
|
|
else
|
|
|
|
S.Diag(Kind.getLocation(), diag::err_ref_init_ambiguous)
|
|
|
|
<< DestType << Args[0]->getType()
|
|
|
|
<< Args[0]->getSourceRange();
|
|
|
|
|
2013-05-03 23:05:50 +08:00
|
|
|
FailedCandidateSet.NoteCandidates(S, OCD_ViableCandidates, Args);
|
2009-12-10 07:02:17 +08:00
|
|
|
break;
|
2011-01-27 15:10:08 +08:00
|
|
|
|
2009-12-10 07:02:17 +08:00
|
|
|
case OR_No_Viable_Function:
|
2013-06-27 11:36:30 +08:00
|
|
|
if (!S.RequireCompleteType(Kind.getLocation(),
|
2013-06-27 09:50:25 +08:00
|
|
|
DestType.getNonReferenceType(),
|
|
|
|
diag::err_typecheck_nonviable_condition_incomplete,
|
|
|
|
Args[0]->getType(), Args[0]->getSourceRange()))
|
|
|
|
S.Diag(Kind.getLocation(), diag::err_typecheck_nonviable_condition)
|
|
|
|
<< Args[0]->getType() << Args[0]->getSourceRange()
|
|
|
|
<< DestType.getNonReferenceType();
|
|
|
|
|
2013-05-03 23:05:50 +08:00
|
|
|
FailedCandidateSet.NoteCandidates(S, OCD_AllCandidates, Args);
|
2009-12-10 07:02:17 +08:00
|
|
|
break;
|
2011-01-27 15:10:08 +08:00
|
|
|
|
2009-12-10 07:02:17 +08:00
|
|
|
case OR_Deleted: {
|
|
|
|
S.Diag(Kind.getLocation(), diag::err_typecheck_deleted_function)
|
|
|
|
<< Args[0]->getType() << DestType.getNonReferenceType()
|
|
|
|
<< Args[0]->getSourceRange();
|
|
|
|
OverloadCandidateSet::iterator Best;
|
2010-08-25 04:38:10 +08:00
|
|
|
OverloadingResult Ovl
|
2010-09-12 16:07:23 +08:00
|
|
|
= FailedCandidateSet.BestViableFunction(S, Kind.getLocation(), Best,
|
|
|
|
true);
|
2009-12-10 07:02:17 +08:00
|
|
|
if (Ovl == OR_Deleted) {
|
2012-03-31 04:53:28 +08:00
|
|
|
S.NoteDeletedFunction(Best->Function);
|
2009-12-10 07:02:17 +08:00
|
|
|
} else {
|
2009-12-12 13:05:38 +08:00
|
|
|
llvm_unreachable("Inconsistent overload resolution?");
|
2009-12-10 07:02:17 +08:00
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
2011-01-27 15:10:08 +08:00
|
|
|
|
2009-12-10 07:02:17 +08:00
|
|
|
case OR_Success:
|
2009-12-12 13:05:38 +08:00
|
|
|
llvm_unreachable("Conversion did not fail!");
|
2009-12-10 07:02:17 +08:00
|
|
|
}
|
|
|
|
break;
|
2011-01-27 15:10:08 +08:00
|
|
|
|
2009-12-10 07:02:17 +08:00
|
|
|
case FK_NonConstLValueReferenceBindingToTemporary:
|
2011-11-28 00:50:07 +08:00
|
|
|
if (isa<InitListExpr>(Args[0])) {
|
|
|
|
S.Diag(Kind.getLocation(),
|
|
|
|
diag::err_lvalue_reference_bind_to_initlist)
|
|
|
|
<< DestType.getNonReferenceType().isVolatileQualified()
|
|
|
|
<< DestType.getNonReferenceType()
|
|
|
|
<< Args[0]->getSourceRange();
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
// Intentional fallthrough
|
|
|
|
|
2009-12-10 07:02:17 +08:00
|
|
|
case FK_NonConstLValueReferenceBindingToUnrelated:
|
2011-01-27 15:10:08 +08:00
|
|
|
S.Diag(Kind.getLocation(),
|
2009-12-10 07:02:17 +08:00
|
|
|
Failure == FK_NonConstLValueReferenceBindingToTemporary
|
|
|
|
? diag::err_lvalue_reference_bind_to_temporary
|
|
|
|
: diag::err_lvalue_reference_bind_to_unrelated)
|
2010-01-30 03:39:15 +08:00
|
|
|
<< DestType.getNonReferenceType().isVolatileQualified()
|
2009-12-10 07:02:17 +08:00
|
|
|
<< DestType.getNonReferenceType()
|
|
|
|
<< Args[0]->getType()
|
|
|
|
<< Args[0]->getSourceRange();
|
|
|
|
break;
|
2011-01-27 15:10:08 +08:00
|
|
|
|
2009-12-10 07:02:17 +08:00
|
|
|
case FK_RValueReferenceBindingToLValue:
|
|
|
|
S.Diag(Kind.getLocation(), diag::err_lvalue_to_rvalue_ref)
|
2011-01-21 09:04:33 +08:00
|
|
|
<< DestType.getNonReferenceType() << Args[0]->getType()
|
2009-12-10 07:02:17 +08:00
|
|
|
<< Args[0]->getSourceRange();
|
|
|
|
break;
|
2011-01-27 15:10:08 +08:00
|
|
|
|
2009-12-10 07:02:17 +08:00
|
|
|
case FK_ReferenceInitDropsQualifiers:
|
|
|
|
S.Diag(Kind.getLocation(), diag::err_reference_bind_drops_quals)
|
|
|
|
<< DestType.getNonReferenceType()
|
|
|
|
<< Args[0]->getType()
|
|
|
|
<< Args[0]->getSourceRange();
|
|
|
|
break;
|
2011-01-27 15:10:08 +08:00
|
|
|
|
2009-12-10 07:02:17 +08:00
|
|
|
case FK_ReferenceInitFailed:
|
|
|
|
S.Diag(Kind.getLocation(), diag::err_reference_bind_failed)
|
|
|
|
<< DestType.getNonReferenceType()
|
2010-11-24 13:12:34 +08:00
|
|
|
<< Args[0]->isLValue()
|
2009-12-10 07:02:17 +08:00
|
|
|
<< Args[0]->getType()
|
|
|
|
<< Args[0]->getSourceRange();
|
2013-03-19 15:04:25 +08:00
|
|
|
emitBadConversionNotes(S, Entity, Args[0]);
|
2009-12-10 07:02:17 +08:00
|
|
|
break;
|
2011-01-27 15:10:08 +08:00
|
|
|
|
2011-02-20 05:32:49 +08:00
|
|
|
case FK_ConversionFailed: {
|
|
|
|
QualType FromType = Args[0]->getType();
|
2011-11-24 06:32:32 +08:00
|
|
|
PartialDiagnostic PDiag = S.PDiag(diag::err_init_conversion_failed)
|
Switch the initialization required by return statements over to the
new InitializationSequence. This fixes some bugs (e.g., PR5808),
changed some diagnostics, and caused more churn than expected. What's
new:
- InitializationSequence now has a "C conversion sequence" category
and step kind, which falls back to
- Changed the diagnostics for returns to always have the result type
of the function first and the type of the expression second.
CheckSingleAssignmentConstraints to peform checking in C.
- Improved ASTs for initialization of return values. The ASTs now
capture all of the temporaries we need to create, but
intentionally do not bind the tempoary that is actually returned,
so that it won't get destroyed twice.
- Make sure to perform an (elidable!) copy of the class object that
is returned from a class.
- Fix copy elision in CodeGen to properly see through the
subexpressions that occur with elidable copies.
- Give "new" its own entity kind; as with return values and thrown
objects, we don't bind the expression so we don't call a
destructor for it.
Note that, with this patch, I've broken returning move-only types in
C++0x. We'll fix it later, when we tackle NRVO.
llvm-svn: 91669
2009-12-18 13:02:21 +08:00
|
|
|
<< (int)Entity.getKind()
|
2009-12-10 07:02:17 +08:00
|
|
|
<< DestType
|
2010-11-24 13:12:34 +08:00
|
|
|
<< Args[0]->isLValue()
|
2011-02-20 05:32:49 +08:00
|
|
|
<< FromType
|
2009-12-10 07:02:17 +08:00
|
|
|
<< Args[0]->getSourceRange();
|
2011-11-24 06:32:32 +08:00
|
|
|
S.HandleFunctionTypeMismatch(PDiag, FromType, DestType);
|
|
|
|
S.Diag(Kind.getLocation(), PDiag);
|
2013-03-19 15:04:25 +08:00
|
|
|
emitBadConversionNotes(S, Entity, Args[0]);
|
2009-12-11 01:56:55 +08:00
|
|
|
break;
|
2011-02-20 05:32:49 +08:00
|
|
|
}
|
2011-04-09 02:41:53 +08:00
|
|
|
|
|
|
|
case FK_ConversionFromPropertyFailed:
|
|
|
|
// No-op. This error has already been reported.
|
|
|
|
break;
|
|
|
|
|
2009-12-11 01:56:55 +08:00
|
|
|
case FK_TooManyInitsForScalar: {
|
2009-12-16 09:38:02 +08:00
|
|
|
SourceRange R;
|
|
|
|
|
|
|
|
if (InitListExpr *InitList = dyn_cast<InitListExpr>(Args[0]))
|
2010-09-09 05:40:08 +08:00
|
|
|
R = SourceRange(InitList->getInit(0)->getLocEnd(),
|
2009-12-16 09:38:02 +08:00
|
|
|
InitList->getLocEnd());
|
2011-01-27 15:10:08 +08:00
|
|
|
else
|
2013-05-03 23:05:50 +08:00
|
|
|
R = SourceRange(Args.front()->getLocEnd(), Args.back()->getLocEnd());
|
2009-12-11 01:56:55 +08:00
|
|
|
|
2010-09-09 05:40:08 +08:00
|
|
|
R.setBegin(S.PP.getLocForEndOfToken(R.getBegin()));
|
|
|
|
if (Kind.isCStyleOrFunctionalCast())
|
|
|
|
S.Diag(Kind.getLocation(), diag::err_builtin_func_cast_more_than_one_arg)
|
|
|
|
<< R;
|
|
|
|
else
|
|
|
|
S.Diag(Kind.getLocation(), diag::err_excess_initializers)
|
|
|
|
<< /*scalar=*/2 << R;
|
2009-12-11 01:56:55 +08:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
case FK_ReferenceBindingToInitList:
|
|
|
|
S.Diag(Kind.getLocation(), diag::err_reference_bind_init_list)
|
|
|
|
<< DestType.getNonReferenceType() << Args[0]->getSourceRange();
|
|
|
|
break;
|
|
|
|
|
|
|
|
case FK_InitListBadDestinationType:
|
|
|
|
S.Diag(Kind.getLocation(), diag::err_init_list_bad_dest_type)
|
|
|
|
<< (DestType->isRecordType()) << DestType << Args[0]->getSourceRange();
|
|
|
|
break;
|
2011-01-27 15:10:08 +08:00
|
|
|
|
2011-12-23 02:58:38 +08:00
|
|
|
case FK_ListConstructorOverloadFailed:
|
2009-12-15 04:49:26 +08:00
|
|
|
case FK_ConstructorOverloadFailed: {
|
|
|
|
SourceRange ArgsRange;
|
2013-05-03 23:05:50 +08:00
|
|
|
if (Args.size())
|
|
|
|
ArgsRange = SourceRange(Args.front()->getLocStart(),
|
|
|
|
Args.back()->getLocEnd());
|
2011-01-27 15:10:08 +08:00
|
|
|
|
2011-12-23 02:58:38 +08:00
|
|
|
if (Failure == FK_ListConstructorOverloadFailed) {
|
2013-05-03 23:05:50 +08:00
|
|
|
assert(Args.size() == 1 && "List construction from other than 1 argument.");
|
2011-12-23 02:58:38 +08:00
|
|
|
InitListExpr *InitList = cast<InitListExpr>(Args[0]);
|
2013-05-03 23:05:50 +08:00
|
|
|
Args = MultiExprArg(InitList->getInits(), InitList->getNumInits());
|
2011-12-23 02:58:38 +08:00
|
|
|
}
|
|
|
|
|
2009-12-15 04:49:26 +08:00
|
|
|
// FIXME: Using "DestType" for the entity we're printing is probably
|
|
|
|
// bad.
|
|
|
|
switch (FailedOverloadResult) {
|
|
|
|
case OR_Ambiguous:
|
|
|
|
S.Diag(Kind.getLocation(), diag::err_ovl_ambiguous_init)
|
|
|
|
<< DestType << ArgsRange;
|
2013-05-03 23:05:50 +08:00
|
|
|
FailedCandidateSet.NoteCandidates(S, OCD_ViableCandidates, Args);
|
2009-12-15 04:49:26 +08:00
|
|
|
break;
|
2011-01-27 15:10:08 +08:00
|
|
|
|
2009-12-15 04:49:26 +08:00
|
|
|
case OR_No_Viable_Function:
|
Rework base and member initialization in constructors, with several
(necessarily simultaneous) changes:
- CXXBaseOrMemberInitializer now contains only a single initializer
rather than a set of initialiation arguments + a constructor. The
single initializer covers all aspects of initialization, including
constructor calls as necessary but also cleanup of temporaries
created by the initializer (which we never handled
before!).
- Rework + simplify code generation for CXXBaseOrMemberInitializers,
since we can now just emit the initializer as an initializer.
- Switched base and member initialization over to the new
initialization code (InitializationSequence), so that it
- Improved diagnostics for the new initialization code when
initializing bases and members, to match the diagnostics produced
by the previous (special-purpose) code.
- Simplify the representation of type-checked constructor initializers in
templates; instead of keeping the fully-type-checked AST, which is
rather hard to undo at template instantiation time, throw away the
type-checked AST and store the raw expressions in the AST. This
simplifies instantiation, but loses a little but of information in
the AST.
- When type-checking implicit base or member initializers within a
dependent context, don't add the generated initializers into the
AST, because they'll look like they were explicit.
- Record in CXXConstructExpr when the constructor call is to
initialize a base class, so that CodeGen does not have to infer it
from context. This ensures that we call the right kind of
constructor.
There are also a few "opportunity" fixes here that were needed to not
regress, for example:
- Diagnose default-initialization of a const-qualified class that
does not have a user-declared default constructor. We had this
diagnostic specifically for bases and members, but missed it for
variables. That's fixed now.
- When defining the implicit constructors, destructor, and
copy-assignment operator, set the CurContext to that constructor
when we're defining the body.
llvm-svn: 94952
2010-01-31 17:12:51 +08:00
|
|
|
if (Kind.getKind() == InitializationKind::IK_Default &&
|
|
|
|
(Entity.getKind() == InitializedEntity::EK_Base ||
|
|
|
|
Entity.getKind() == InitializedEntity::EK_Member) &&
|
|
|
|
isa<CXXConstructorDecl>(S.CurContext)) {
|
|
|
|
// This is implicit default initialization of a member or
|
|
|
|
// base within a constructor. If no viable function was
|
|
|
|
// found, notify the user that she needs to explicitly
|
|
|
|
// initialize this base/member.
|
|
|
|
CXXConstructorDecl *Constructor
|
|
|
|
= cast<CXXConstructorDecl>(S.CurContext);
|
|
|
|
if (Entity.getKind() == InitializedEntity::EK_Base) {
|
|
|
|
S.Diag(Kind.getLocation(), diag::err_missing_default_ctor)
|
2013-03-19 05:12:30 +08:00
|
|
|
<< (Constructor->getInheritedConstructor() ? 2 :
|
|
|
|
Constructor->isImplicit() ? 1 : 0)
|
Rework base and member initialization in constructors, with several
(necessarily simultaneous) changes:
- CXXBaseOrMemberInitializer now contains only a single initializer
rather than a set of initialiation arguments + a constructor. The
single initializer covers all aspects of initialization, including
constructor calls as necessary but also cleanup of temporaries
created by the initializer (which we never handled
before!).
- Rework + simplify code generation for CXXBaseOrMemberInitializers,
since we can now just emit the initializer as an initializer.
- Switched base and member initialization over to the new
initialization code (InitializationSequence), so that it
- Improved diagnostics for the new initialization code when
initializing bases and members, to match the diagnostics produced
by the previous (special-purpose) code.
- Simplify the representation of type-checked constructor initializers in
templates; instead of keeping the fully-type-checked AST, which is
rather hard to undo at template instantiation time, throw away the
type-checked AST and store the raw expressions in the AST. This
simplifies instantiation, but loses a little but of information in
the AST.
- When type-checking implicit base or member initializers within a
dependent context, don't add the generated initializers into the
AST, because they'll look like they were explicit.
- Record in CXXConstructExpr when the constructor call is to
initialize a base class, so that CodeGen does not have to infer it
from context. This ensures that we call the right kind of
constructor.
There are also a few "opportunity" fixes here that were needed to not
regress, for example:
- Diagnose default-initialization of a const-qualified class that
does not have a user-declared default constructor. We had this
diagnostic specifically for bases and members, but missed it for
variables. That's fixed now.
- When defining the implicit constructors, destructor, and
copy-assignment operator, set the CurContext to that constructor
when we're defining the body.
llvm-svn: 94952
2010-01-31 17:12:51 +08:00
|
|
|
<< S.Context.getTypeDeclType(Constructor->getParent())
|
|
|
|
<< /*base=*/0
|
|
|
|
<< Entity.getType();
|
|
|
|
|
|
|
|
RecordDecl *BaseDecl
|
|
|
|
= Entity.getBaseSpecifier()->getType()->getAs<RecordType>()
|
|
|
|
->getDecl();
|
|
|
|
S.Diag(BaseDecl->getLocation(), diag::note_previous_decl)
|
|
|
|
<< S.Context.getTagDeclType(BaseDecl);
|
|
|
|
} else {
|
|
|
|
S.Diag(Kind.getLocation(), diag::err_missing_default_ctor)
|
2013-03-19 05:12:30 +08:00
|
|
|
<< (Constructor->getInheritedConstructor() ? 2 :
|
|
|
|
Constructor->isImplicit() ? 1 : 0)
|
Rework base and member initialization in constructors, with several
(necessarily simultaneous) changes:
- CXXBaseOrMemberInitializer now contains only a single initializer
rather than a set of initialiation arguments + a constructor. The
single initializer covers all aspects of initialization, including
constructor calls as necessary but also cleanup of temporaries
created by the initializer (which we never handled
before!).
- Rework + simplify code generation for CXXBaseOrMemberInitializers,
since we can now just emit the initializer as an initializer.
- Switched base and member initialization over to the new
initialization code (InitializationSequence), so that it
- Improved diagnostics for the new initialization code when
initializing bases and members, to match the diagnostics produced
by the previous (special-purpose) code.
- Simplify the representation of type-checked constructor initializers in
templates; instead of keeping the fully-type-checked AST, which is
rather hard to undo at template instantiation time, throw away the
type-checked AST and store the raw expressions in the AST. This
simplifies instantiation, but loses a little but of information in
the AST.
- When type-checking implicit base or member initializers within a
dependent context, don't add the generated initializers into the
AST, because they'll look like they were explicit.
- Record in CXXConstructExpr when the constructor call is to
initialize a base class, so that CodeGen does not have to infer it
from context. This ensures that we call the right kind of
constructor.
There are also a few "opportunity" fixes here that were needed to not
regress, for example:
- Diagnose default-initialization of a const-qualified class that
does not have a user-declared default constructor. We had this
diagnostic specifically for bases and members, but missed it for
variables. That's fixed now.
- When defining the implicit constructors, destructor, and
copy-assignment operator, set the CurContext to that constructor
when we're defining the body.
llvm-svn: 94952
2010-01-31 17:12:51 +08:00
|
|
|
<< S.Context.getTypeDeclType(Constructor->getParent())
|
|
|
|
<< /*member=*/1
|
|
|
|
<< Entity.getName();
|
|
|
|
S.Diag(Entity.getDecl()->getLocation(), diag::note_field_decl);
|
|
|
|
|
|
|
|
if (const RecordType *Record
|
|
|
|
= Entity.getType()->getAs<RecordType>())
|
2011-01-27 15:10:08 +08:00
|
|
|
S.Diag(Record->getDecl()->getLocation(),
|
Rework base and member initialization in constructors, with several
(necessarily simultaneous) changes:
- CXXBaseOrMemberInitializer now contains only a single initializer
rather than a set of initialiation arguments + a constructor. The
single initializer covers all aspects of initialization, including
constructor calls as necessary but also cleanup of temporaries
created by the initializer (which we never handled
before!).
- Rework + simplify code generation for CXXBaseOrMemberInitializers,
since we can now just emit the initializer as an initializer.
- Switched base and member initialization over to the new
initialization code (InitializationSequence), so that it
- Improved diagnostics for the new initialization code when
initializing bases and members, to match the diagnostics produced
by the previous (special-purpose) code.
- Simplify the representation of type-checked constructor initializers in
templates; instead of keeping the fully-type-checked AST, which is
rather hard to undo at template instantiation time, throw away the
type-checked AST and store the raw expressions in the AST. This
simplifies instantiation, but loses a little but of information in
the AST.
- When type-checking implicit base or member initializers within a
dependent context, don't add the generated initializers into the
AST, because they'll look like they were explicit.
- Record in CXXConstructExpr when the constructor call is to
initialize a base class, so that CodeGen does not have to infer it
from context. This ensures that we call the right kind of
constructor.
There are also a few "opportunity" fixes here that were needed to not
regress, for example:
- Diagnose default-initialization of a const-qualified class that
does not have a user-declared default constructor. We had this
diagnostic specifically for bases and members, but missed it for
variables. That's fixed now.
- When defining the implicit constructors, destructor, and
copy-assignment operator, set the CurContext to that constructor
when we're defining the body.
llvm-svn: 94952
2010-01-31 17:12:51 +08:00
|
|
|
diag::note_previous_decl)
|
|
|
|
<< S.Context.getTagDeclType(Record->getDecl());
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2009-12-15 04:49:26 +08:00
|
|
|
S.Diag(Kind.getLocation(), diag::err_ovl_no_viable_function_in_init)
|
|
|
|
<< DestType << ArgsRange;
|
2013-05-03 23:05:50 +08:00
|
|
|
FailedCandidateSet.NoteCandidates(S, OCD_AllCandidates, Args);
|
2009-12-15 04:49:26 +08:00
|
|
|
break;
|
2011-01-27 15:10:08 +08:00
|
|
|
|
2009-12-15 04:49:26 +08:00
|
|
|
case OR_Deleted: {
|
|
|
|
OverloadCandidateSet::iterator Best;
|
2010-08-25 04:38:10 +08:00
|
|
|
OverloadingResult Ovl
|
|
|
|
= FailedCandidateSet.BestViableFunction(S, Kind.getLocation(), Best);
|
2012-02-16 03:33:52 +08:00
|
|
|
if (Ovl != OR_Deleted) {
|
|
|
|
S.Diag(Kind.getLocation(), diag::err_ovl_deleted_init)
|
|
|
|
<< true << DestType << ArgsRange;
|
2009-12-15 04:49:26 +08:00
|
|
|
llvm_unreachable("Inconsistent overload resolution?");
|
2012-02-16 03:33:52 +08:00
|
|
|
break;
|
2009-12-15 04:49:26 +08:00
|
|
|
}
|
2012-02-16 03:33:52 +08:00
|
|
|
|
|
|
|
// If this is a defaulted or implicitly-declared function, then
|
|
|
|
// it was implicitly deleted. Make it clear that the deletion was
|
|
|
|
// implicit.
|
2012-03-31 04:53:28 +08:00
|
|
|
if (S.isImplicitlyDeleted(Best->Function))
|
2012-02-16 03:33:52 +08:00
|
|
|
S.Diag(Kind.getLocation(), diag::err_ovl_deleted_special_init)
|
2012-03-31 04:53:28 +08:00
|
|
|
<< S.getSpecialMember(cast<CXXMethodDecl>(Best->Function))
|
2012-02-16 03:33:52 +08:00
|
|
|
<< DestType << ArgsRange;
|
2012-03-31 04:53:28 +08:00
|
|
|
else
|
|
|
|
S.Diag(Kind.getLocation(), diag::err_ovl_deleted_init)
|
|
|
|
<< true << DestType << ArgsRange;
|
|
|
|
|
|
|
|
S.NoteDeletedFunction(Best->Function);
|
2009-12-15 04:49:26 +08:00
|
|
|
break;
|
|
|
|
}
|
2011-01-27 15:10:08 +08:00
|
|
|
|
2009-12-15 04:49:26 +08:00
|
|
|
case OR_Success:
|
|
|
|
llvm_unreachable("Conversion did not fail!");
|
|
|
|
}
|
|
|
|
}
|
2012-01-17 16:24:58 +08:00
|
|
|
break;
|
2011-01-27 15:10:08 +08:00
|
|
|
|
2009-12-16 09:38:02 +08:00
|
|
|
case FK_DefaultInitOfConst:
|
Rework base and member initialization in constructors, with several
(necessarily simultaneous) changes:
- CXXBaseOrMemberInitializer now contains only a single initializer
rather than a set of initialiation arguments + a constructor. The
single initializer covers all aspects of initialization, including
constructor calls as necessary but also cleanup of temporaries
created by the initializer (which we never handled
before!).
- Rework + simplify code generation for CXXBaseOrMemberInitializers,
since we can now just emit the initializer as an initializer.
- Switched base and member initialization over to the new
initialization code (InitializationSequence), so that it
- Improved diagnostics for the new initialization code when
initializing bases and members, to match the diagnostics produced
by the previous (special-purpose) code.
- Simplify the representation of type-checked constructor initializers in
templates; instead of keeping the fully-type-checked AST, which is
rather hard to undo at template instantiation time, throw away the
type-checked AST and store the raw expressions in the AST. This
simplifies instantiation, but loses a little but of information in
the AST.
- When type-checking implicit base or member initializers within a
dependent context, don't add the generated initializers into the
AST, because they'll look like they were explicit.
- Record in CXXConstructExpr when the constructor call is to
initialize a base class, so that CodeGen does not have to infer it
from context. This ensures that we call the right kind of
constructor.
There are also a few "opportunity" fixes here that were needed to not
regress, for example:
- Diagnose default-initialization of a const-qualified class that
does not have a user-declared default constructor. We had this
diagnostic specifically for bases and members, but missed it for
variables. That's fixed now.
- When defining the implicit constructors, destructor, and
copy-assignment operator, set the CurContext to that constructor
when we're defining the body.
llvm-svn: 94952
2010-01-31 17:12:51 +08:00
|
|
|
if (Entity.getKind() == InitializedEntity::EK_Member &&
|
|
|
|
isa<CXXConstructorDecl>(S.CurContext)) {
|
|
|
|
// This is implicit default-initialization of a const member in
|
|
|
|
// a constructor. Complain that it needs to be explicitly
|
|
|
|
// initialized.
|
|
|
|
CXXConstructorDecl *Constructor = cast<CXXConstructorDecl>(S.CurContext);
|
|
|
|
S.Diag(Kind.getLocation(), diag::err_uninitialized_member_in_ctor)
|
2013-03-19 05:12:30 +08:00
|
|
|
<< (Constructor->getInheritedConstructor() ? 2 :
|
|
|
|
Constructor->isImplicit() ? 1 : 0)
|
Rework base and member initialization in constructors, with several
(necessarily simultaneous) changes:
- CXXBaseOrMemberInitializer now contains only a single initializer
rather than a set of initialiation arguments + a constructor. The
single initializer covers all aspects of initialization, including
constructor calls as necessary but also cleanup of temporaries
created by the initializer (which we never handled
before!).
- Rework + simplify code generation for CXXBaseOrMemberInitializers,
since we can now just emit the initializer as an initializer.
- Switched base and member initialization over to the new
initialization code (InitializationSequence), so that it
- Improved diagnostics for the new initialization code when
initializing bases and members, to match the diagnostics produced
by the previous (special-purpose) code.
- Simplify the representation of type-checked constructor initializers in
templates; instead of keeping the fully-type-checked AST, which is
rather hard to undo at template instantiation time, throw away the
type-checked AST and store the raw expressions in the AST. This
simplifies instantiation, but loses a little but of information in
the AST.
- When type-checking implicit base or member initializers within a
dependent context, don't add the generated initializers into the
AST, because they'll look like they were explicit.
- Record in CXXConstructExpr when the constructor call is to
initialize a base class, so that CodeGen does not have to infer it
from context. This ensures that we call the right kind of
constructor.
There are also a few "opportunity" fixes here that were needed to not
regress, for example:
- Diagnose default-initialization of a const-qualified class that
does not have a user-declared default constructor. We had this
diagnostic specifically for bases and members, but missed it for
variables. That's fixed now.
- When defining the implicit constructors, destructor, and
copy-assignment operator, set the CurContext to that constructor
when we're defining the body.
llvm-svn: 94952
2010-01-31 17:12:51 +08:00
|
|
|
<< S.Context.getTypeDeclType(Constructor->getParent())
|
|
|
|
<< /*const=*/1
|
|
|
|
<< Entity.getName();
|
|
|
|
S.Diag(Entity.getDecl()->getLocation(), diag::note_previous_decl)
|
|
|
|
<< Entity.getName();
|
|
|
|
} else {
|
|
|
|
S.Diag(Kind.getLocation(), diag::err_default_init_const)
|
|
|
|
<< DestType << (bool)DestType->getAs<RecordType>();
|
|
|
|
}
|
2009-12-16 09:38:02 +08:00
|
|
|
break;
|
2011-01-27 15:10:08 +08:00
|
|
|
|
2011-09-25 01:47:52 +08:00
|
|
|
case FK_Incomplete:
|
2012-04-11 04:43:46 +08:00
|
|
|
S.RequireCompleteType(Kind.getLocation(), FailedIncompleteType,
|
2011-09-25 01:47:52 +08:00
|
|
|
diag::err_init_incomplete_type);
|
|
|
|
break;
|
|
|
|
|
2011-09-25 01:48:00 +08:00
|
|
|
case FK_ListInitializationFailed: {
|
|
|
|
// Run the init list checker again to emit diagnostics.
|
2013-11-22 07:30:57 +08:00
|
|
|
InitListExpr *InitList = cast<InitListExpr>(Args[0]);
|
|
|
|
diagnoseListInit(S, Entity, InitList);
|
2011-09-25 01:48:00 +08:00
|
|
|
break;
|
|
|
|
}
|
2011-10-18 02:40:02 +08:00
|
|
|
|
|
|
|
case FK_PlaceholderType: {
|
|
|
|
// FIXME: Already diagnosed!
|
|
|
|
break;
|
|
|
|
}
|
2012-01-18 06:49:42 +08:00
|
|
|
|
2012-04-02 03:54:59 +08:00
|
|
|
case FK_ExplicitConstructor: {
|
|
|
|
S.Diag(Kind.getLocation(), diag::err_selected_explicit_constructor)
|
|
|
|
<< Args[0]->getSourceRange();
|
|
|
|
OverloadCandidateSet::iterator Best;
|
|
|
|
OverloadingResult Ovl
|
|
|
|
= FailedCandidateSet.BestViableFunction(S, Kind.getLocation(), Best);
|
2012-04-03 03:05:35 +08:00
|
|
|
(void)Ovl;
|
2012-04-02 03:54:59 +08:00
|
|
|
assert(Ovl == OR_Success && "Inconsistent overload resolution");
|
|
|
|
CXXConstructorDecl *CtorDecl = cast<CXXConstructorDecl>(Best->Function);
|
|
|
|
S.Diag(CtorDecl->getLocation(), diag::note_constructor_declared_here);
|
|
|
|
break;
|
|
|
|
}
|
2009-12-10 07:02:17 +08:00
|
|
|
}
|
2011-01-27 15:10:08 +08:00
|
|
|
|
2010-04-22 08:20:18 +08:00
|
|
|
PrintInitLocationNote(S, Entity);
|
2009-12-10 07:02:17 +08:00
|
|
|
return true;
|
|
|
|
}
|
Switch the initialization required by return statements over to the
new InitializationSequence. This fixes some bugs (e.g., PR5808),
changed some diagnostics, and caused more churn than expected. What's
new:
- InitializationSequence now has a "C conversion sequence" category
and step kind, which falls back to
- Changed the diagnostics for returns to always have the result type
of the function first and the type of the expression second.
CheckSingleAssignmentConstraints to peform checking in C.
- Improved ASTs for initialization of return values. The ASTs now
capture all of the temporaries we need to create, but
intentionally do not bind the tempoary that is actually returned,
so that it won't get destroyed twice.
- Make sure to perform an (elidable!) copy of the class object that
is returned from a class.
- Fix copy elision in CodeGen to properly see through the
subexpressions that occur with elidable copies.
- Give "new" its own entity kind; as with return values and thrown
objects, we don't bind the expression so we don't call a
destructor for it.
Note that, with this patch, I've broken returning move-only types in
C++0x. We'll fix it later, when we tackle NRVO.
llvm-svn: 91669
2009-12-18 13:02:21 +08:00
|
|
|
|
2011-07-23 18:55:15 +08:00
|
|
|
void InitializationSequence::dump(raw_ostream &OS) const {
|
2010-01-30 03:14:02 +08:00
|
|
|
switch (SequenceKind) {
|
|
|
|
case FailedSequence: {
|
|
|
|
OS << "Failed sequence: ";
|
|
|
|
switch (Failure) {
|
|
|
|
case FK_TooManyInitsForReference:
|
|
|
|
OS << "too many initializers for reference";
|
|
|
|
break;
|
2011-01-27 15:10:08 +08:00
|
|
|
|
2010-01-30 03:14:02 +08:00
|
|
|
case FK_ArrayNeedsInitList:
|
|
|
|
OS << "array requires initializer list";
|
|
|
|
break;
|
2011-01-27 15:10:08 +08:00
|
|
|
|
2010-01-30 03:14:02 +08:00
|
|
|
case FK_ArrayNeedsInitListOrStringLiteral:
|
|
|
|
OS << "array requires initializer list or string literal";
|
|
|
|
break;
|
2011-01-27 15:10:08 +08:00
|
|
|
|
2013-05-15 19:03:04 +08:00
|
|
|
case FK_ArrayNeedsInitListOrWideStringLiteral:
|
|
|
|
OS << "array requires initializer list or wide string literal";
|
|
|
|
break;
|
|
|
|
|
|
|
|
case FK_NarrowStringIntoWideCharArray:
|
|
|
|
OS << "narrow string into wide char array";
|
|
|
|
break;
|
|
|
|
|
|
|
|
case FK_WideStringIntoCharArray:
|
|
|
|
OS << "wide string into char array";
|
|
|
|
break;
|
|
|
|
|
|
|
|
case FK_IncompatWideStringIntoWideChar:
|
|
|
|
OS << "incompatible wide string into wide char array";
|
|
|
|
break;
|
|
|
|
|
2011-02-23 02:29:51 +08:00
|
|
|
case FK_ArrayTypeMismatch:
|
|
|
|
OS << "array type mismatch";
|
|
|
|
break;
|
|
|
|
|
|
|
|
case FK_NonConstantArrayInit:
|
|
|
|
OS << "non-constant array initializer";
|
|
|
|
break;
|
|
|
|
|
2010-01-30 03:14:02 +08:00
|
|
|
case FK_AddressOfOverloadFailed:
|
|
|
|
OS << "address of overloaded function failed";
|
|
|
|
break;
|
2011-01-27 15:10:08 +08:00
|
|
|
|
2010-01-30 03:14:02 +08:00
|
|
|
case FK_ReferenceInitOverloadFailed:
|
|
|
|
OS << "overload resolution for reference initialization failed";
|
|
|
|
break;
|
2011-01-27 15:10:08 +08:00
|
|
|
|
2010-01-30 03:14:02 +08:00
|
|
|
case FK_NonConstLValueReferenceBindingToTemporary:
|
|
|
|
OS << "non-const lvalue reference bound to temporary";
|
|
|
|
break;
|
2011-01-27 15:10:08 +08:00
|
|
|
|
2010-01-30 03:14:02 +08:00
|
|
|
case FK_NonConstLValueReferenceBindingToUnrelated:
|
|
|
|
OS << "non-const lvalue reference bound to unrelated type";
|
|
|
|
break;
|
2011-01-27 15:10:08 +08:00
|
|
|
|
2010-01-30 03:14:02 +08:00
|
|
|
case FK_RValueReferenceBindingToLValue:
|
|
|
|
OS << "rvalue reference bound to an lvalue";
|
|
|
|
break;
|
2011-01-27 15:10:08 +08:00
|
|
|
|
2010-01-30 03:14:02 +08:00
|
|
|
case FK_ReferenceInitDropsQualifiers:
|
|
|
|
OS << "reference initialization drops qualifiers";
|
|
|
|
break;
|
2011-01-27 15:10:08 +08:00
|
|
|
|
2010-01-30 03:14:02 +08:00
|
|
|
case FK_ReferenceInitFailed:
|
|
|
|
OS << "reference initialization failed";
|
|
|
|
break;
|
2011-01-27 15:10:08 +08:00
|
|
|
|
2010-01-30 03:14:02 +08:00
|
|
|
case FK_ConversionFailed:
|
|
|
|
OS << "conversion failed";
|
|
|
|
break;
|
2011-01-27 15:10:08 +08:00
|
|
|
|
2011-04-09 02:41:53 +08:00
|
|
|
case FK_ConversionFromPropertyFailed:
|
|
|
|
OS << "conversion from property failed";
|
|
|
|
break;
|
|
|
|
|
2010-01-30 03:14:02 +08:00
|
|
|
case FK_TooManyInitsForScalar:
|
|
|
|
OS << "too many initializers for scalar";
|
|
|
|
break;
|
2011-01-27 15:10:08 +08:00
|
|
|
|
2010-01-30 03:14:02 +08:00
|
|
|
case FK_ReferenceBindingToInitList:
|
|
|
|
OS << "referencing binding to initializer list";
|
|
|
|
break;
|
2011-01-27 15:10:08 +08:00
|
|
|
|
2010-01-30 03:14:02 +08:00
|
|
|
case FK_InitListBadDestinationType:
|
|
|
|
OS << "initializer list for non-aggregate, non-scalar type";
|
|
|
|
break;
|
2011-01-27 15:10:08 +08:00
|
|
|
|
2010-01-30 03:14:02 +08:00
|
|
|
case FK_UserConversionOverloadFailed:
|
|
|
|
OS << "overloading failed for user-defined conversion";
|
|
|
|
break;
|
2011-01-27 15:10:08 +08:00
|
|
|
|
2010-01-30 03:14:02 +08:00
|
|
|
case FK_ConstructorOverloadFailed:
|
|
|
|
OS << "constructor overloading failed";
|
|
|
|
break;
|
2011-01-27 15:10:08 +08:00
|
|
|
|
2010-01-30 03:14:02 +08:00
|
|
|
case FK_DefaultInitOfConst:
|
|
|
|
OS << "default initialization of a const variable";
|
|
|
|
break;
|
2011-01-27 15:10:08 +08:00
|
|
|
|
2010-05-21 06:12:02 +08:00
|
|
|
case FK_Incomplete:
|
|
|
|
OS << "initialization of incomplete type";
|
|
|
|
break;
|
2011-09-25 01:47:52 +08:00
|
|
|
|
|
|
|
case FK_ListInitializationFailed:
|
2011-09-25 01:48:00 +08:00
|
|
|
OS << "list initialization checker failure";
|
2011-10-18 02:40:02 +08:00
|
|
|
break;
|
|
|
|
|
2012-01-05 08:13:19 +08:00
|
|
|
case FK_VariableLengthArrayHasInitializer:
|
|
|
|
OS << "variable length array has an initializer";
|
|
|
|
break;
|
|
|
|
|
2011-10-18 02:40:02 +08:00
|
|
|
case FK_PlaceholderType:
|
|
|
|
OS << "initializer expression isn't contextually valid";
|
|
|
|
break;
|
2011-12-23 04:21:32 +08:00
|
|
|
|
|
|
|
case FK_ListConstructorOverloadFailed:
|
|
|
|
OS << "list constructor overloading failed";
|
|
|
|
break;
|
2012-01-18 06:49:42 +08:00
|
|
|
|
2012-04-02 03:54:59 +08:00
|
|
|
case FK_ExplicitConstructor:
|
|
|
|
OS << "list copy initialization chose explicit constructor";
|
|
|
|
break;
|
2011-01-27 15:10:08 +08:00
|
|
|
}
|
2010-01-30 03:14:02 +08:00
|
|
|
OS << '\n';
|
|
|
|
return;
|
|
|
|
}
|
2011-01-27 15:10:08 +08:00
|
|
|
|
2010-01-30 03:14:02 +08:00
|
|
|
case DependentSequence:
|
2011-06-05 21:59:11 +08:00
|
|
|
OS << "Dependent sequence\n";
|
2010-01-30 03:14:02 +08:00
|
|
|
return;
|
2011-01-27 15:10:08 +08:00
|
|
|
|
2011-06-05 21:59:11 +08:00
|
|
|
case NormalSequence:
|
|
|
|
OS << "Normal sequence: ";
|
2010-01-30 03:14:02 +08:00
|
|
|
break;
|
|
|
|
}
|
2011-01-27 15:10:08 +08:00
|
|
|
|
2010-01-30 03:14:02 +08:00
|
|
|
for (step_iterator S = step_begin(), SEnd = step_end(); S != SEnd; ++S) {
|
|
|
|
if (S != step_begin()) {
|
|
|
|
OS << " -> ";
|
|
|
|
}
|
2011-01-27 15:10:08 +08:00
|
|
|
|
2010-01-30 03:14:02 +08:00
|
|
|
switch (S->Kind) {
|
|
|
|
case SK_ResolveAddressOfOverloadedFunction:
|
|
|
|
OS << "resolve address of overloaded function";
|
|
|
|
break;
|
2011-01-27 15:10:08 +08:00
|
|
|
|
2010-01-30 03:14:02 +08:00
|
|
|
case SK_CastDerivedToBaseRValue:
|
|
|
|
OS << "derived-to-base case (rvalue" << S->Type.getAsString() << ")";
|
|
|
|
break;
|
2011-01-27 15:10:08 +08:00
|
|
|
|
2010-07-20 12:20:21 +08:00
|
|
|
case SK_CastDerivedToBaseXValue:
|
|
|
|
OS << "derived-to-base case (xvalue" << S->Type.getAsString() << ")";
|
|
|
|
break;
|
2011-01-27 15:10:08 +08:00
|
|
|
|
2010-01-30 03:14:02 +08:00
|
|
|
case SK_CastDerivedToBaseLValue:
|
|
|
|
OS << "derived-to-base case (lvalue" << S->Type.getAsString() << ")";
|
|
|
|
break;
|
2011-01-27 15:10:08 +08:00
|
|
|
|
2010-01-30 03:14:02 +08:00
|
|
|
case SK_BindReference:
|
|
|
|
OS << "bind reference to lvalue";
|
|
|
|
break;
|
2011-01-27 15:10:08 +08:00
|
|
|
|
2010-01-30 03:14:02 +08:00
|
|
|
case SK_BindReferenceToTemporary:
|
|
|
|
OS << "bind reference to a temporary";
|
|
|
|
break;
|
2011-01-27 15:10:08 +08:00
|
|
|
|
2010-04-18 15:40:54 +08:00
|
|
|
case SK_ExtraneousCopyToTemporary:
|
|
|
|
OS << "extraneous C++03 copy to temporary";
|
|
|
|
break;
|
|
|
|
|
2010-01-30 03:14:02 +08:00
|
|
|
case SK_UserConversion:
|
2011-10-15 02:45:37 +08:00
|
|
|
OS << "user-defined conversion via " << *S->Function.Function;
|
2010-01-30 03:14:02 +08:00
|
|
|
break;
|
2010-07-20 12:20:21 +08:00
|
|
|
|
2010-01-30 03:14:02 +08:00
|
|
|
case SK_QualificationConversionRValue:
|
|
|
|
OS << "qualification conversion (rvalue)";
|
2011-11-28 00:50:07 +08:00
|
|
|
break;
|
2010-01-30 03:14:02 +08:00
|
|
|
|
2010-07-20 12:20:21 +08:00
|
|
|
case SK_QualificationConversionXValue:
|
|
|
|
OS << "qualification conversion (xvalue)";
|
2011-11-28 00:50:07 +08:00
|
|
|
break;
|
2010-07-20 12:20:21 +08:00
|
|
|
|
2010-01-30 03:14:02 +08:00
|
|
|
case SK_QualificationConversionLValue:
|
|
|
|
OS << "qualification conversion (lvalue)";
|
|
|
|
break;
|
2011-01-27 15:10:08 +08:00
|
|
|
|
2013-04-11 08:58:58 +08:00
|
|
|
case SK_LValueToRValue:
|
|
|
|
OS << "load (lvalue to rvalue)";
|
|
|
|
break;
|
|
|
|
|
2010-01-30 03:14:02 +08:00
|
|
|
case SK_ConversionSequence:
|
|
|
|
OS << "implicit conversion sequence (";
|
2013-11-08 10:16:10 +08:00
|
|
|
S->ICS->dump(); // FIXME: use OS
|
2010-01-30 03:14:02 +08:00
|
|
|
OS << ")";
|
|
|
|
break;
|
2011-01-27 15:10:08 +08:00
|
|
|
|
2013-09-22 05:19:19 +08:00
|
|
|
case SK_ConversionSequenceNoNarrowing:
|
|
|
|
OS << "implicit conversion sequence with narrowing prohibited (";
|
2013-11-08 10:16:10 +08:00
|
|
|
S->ICS->dump(); // FIXME: use OS
|
2013-09-22 05:19:19 +08:00
|
|
|
OS << ")";
|
|
|
|
break;
|
|
|
|
|
2010-01-30 03:14:02 +08:00
|
|
|
case SK_ListInitialization:
|
2011-09-25 01:47:52 +08:00
|
|
|
OS << "list aggregate initialization";
|
|
|
|
break;
|
|
|
|
|
|
|
|
case SK_ListConstructorCall:
|
|
|
|
OS << "list initialization via constructor";
|
2010-01-30 03:14:02 +08:00
|
|
|
break;
|
2011-01-27 15:10:08 +08:00
|
|
|
|
2011-11-28 00:50:07 +08:00
|
|
|
case SK_UnwrapInitList:
|
|
|
|
OS << "unwrap reference initializer list";
|
|
|
|
break;
|
|
|
|
|
|
|
|
case SK_RewrapInitList:
|
|
|
|
OS << "rewrap reference initializer list";
|
|
|
|
break;
|
|
|
|
|
2010-01-30 03:14:02 +08:00
|
|
|
case SK_ConstructorInitialization:
|
|
|
|
OS << "constructor initialization";
|
|
|
|
break;
|
2011-01-27 15:10:08 +08:00
|
|
|
|
2010-01-30 03:14:02 +08:00
|
|
|
case SK_ZeroInitialization:
|
|
|
|
OS << "zero initialization";
|
|
|
|
break;
|
2011-01-27 15:10:08 +08:00
|
|
|
|
2010-01-30 03:14:02 +08:00
|
|
|
case SK_CAssignment:
|
|
|
|
OS << "C assignment";
|
|
|
|
break;
|
2011-01-27 15:10:08 +08:00
|
|
|
|
2010-01-30 03:14:02 +08:00
|
|
|
case SK_StringInit:
|
|
|
|
OS << "string initialization";
|
|
|
|
break;
|
2010-08-07 19:51:51 +08:00
|
|
|
|
|
|
|
case SK_ObjCObjectConversion:
|
|
|
|
OS << "Objective-C object conversion";
|
|
|
|
break;
|
2011-02-23 02:29:51 +08:00
|
|
|
|
|
|
|
case SK_ArrayInit:
|
|
|
|
OS << "array initialization";
|
|
|
|
break;
|
2011-06-16 07:02:42 +08:00
|
|
|
|
2012-02-16 06:38:09 +08:00
|
|
|
case SK_ParenthesizedArrayInit:
|
|
|
|
OS << "parenthesized array initialization";
|
|
|
|
break;
|
|
|
|
|
2011-06-16 07:02:42 +08:00
|
|
|
case SK_PassByIndirectCopyRestore:
|
|
|
|
OS << "pass by indirect copy and restore";
|
|
|
|
break;
|
|
|
|
|
|
|
|
case SK_PassByIndirectRestore:
|
|
|
|
OS << "pass by indirect restore";
|
|
|
|
break;
|
|
|
|
|
|
|
|
case SK_ProduceObjCObject:
|
|
|
|
OS << "Objective-C object retension";
|
|
|
|
break;
|
2012-01-18 06:49:42 +08:00
|
|
|
|
|
|
|
case SK_StdInitializerList:
|
|
|
|
OS << "std::initializer_list from initializer list";
|
|
|
|
break;
|
2013-01-20 20:31:11 +08:00
|
|
|
|
2013-02-07 18:55:47 +08:00
|
|
|
case SK_OCLSamplerInit:
|
|
|
|
OS << "OpenCL sampler_t from integer constant";
|
|
|
|
break;
|
|
|
|
|
2013-01-20 20:31:11 +08:00
|
|
|
case SK_OCLZeroEvent:
|
|
|
|
OS << "OpenCL event_t from zero";
|
|
|
|
break;
|
2010-01-30 03:14:02 +08:00
|
|
|
}
|
2013-02-05 13:52:24 +08:00
|
|
|
|
|
|
|
OS << " [" << S->Type.getAsString() << ']';
|
2010-01-30 03:14:02 +08:00
|
|
|
}
|
2013-02-05 13:52:24 +08:00
|
|
|
|
|
|
|
OS << '\n';
|
2010-01-30 03:14:02 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
void InitializationSequence::dump() const {
|
|
|
|
dump(llvm::errs());
|
|
|
|
}
|
|
|
|
|
2013-09-22 05:19:19 +08:00
|
|
|
static void DiagnoseNarrowingInInitList(Sema &S,
|
|
|
|
const ImplicitConversionSequence &ICS,
|
|
|
|
QualType PreNarrowingType,
|
2012-01-18 13:21:49 +08:00
|
|
|
QualType EntityType,
|
|
|
|
const Expr *PostInit) {
|
|
|
|
const StandardConversionSequence *SCS = 0;
|
|
|
|
switch (ICS.getKind()) {
|
|
|
|
case ImplicitConversionSequence::StandardConversion:
|
|
|
|
SCS = &ICS.Standard;
|
|
|
|
break;
|
|
|
|
case ImplicitConversionSequence::UserDefinedConversion:
|
|
|
|
SCS = &ICS.UserDefined.After;
|
|
|
|
break;
|
|
|
|
case ImplicitConversionSequence::AmbiguousConversion:
|
|
|
|
case ImplicitConversionSequence::EllipsisConversion:
|
|
|
|
case ImplicitConversionSequence::BadConversion:
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
// C++11 [dcl.init.list]p7: Check whether this is a narrowing conversion.
|
|
|
|
APValue ConstantValue;
|
2012-03-24 07:55:39 +08:00
|
|
|
QualType ConstantType;
|
|
|
|
switch (SCS->getNarrowingKind(S.Context, PostInit, ConstantValue,
|
|
|
|
ConstantType)) {
|
2012-01-18 13:21:49 +08:00
|
|
|
case NK_Not_Narrowing:
|
|
|
|
// No narrowing occurred.
|
|
|
|
return;
|
|
|
|
|
|
|
|
case NK_Type_Narrowing:
|
|
|
|
// This was a floating-to-integer conversion, which is always considered a
|
|
|
|
// narrowing conversion even if the value is a constant and can be
|
|
|
|
// represented exactly as an integer.
|
|
|
|
S.Diag(PostInit->getLocStart(),
|
2013-11-12 10:41:45 +08:00
|
|
|
(S.getLangOpts().MicrosoftExt || !S.getLangOpts().CPlusPlus11)
|
|
|
|
? diag::warn_init_list_type_narrowing
|
|
|
|
: diag::ext_init_list_type_narrowing)
|
2012-01-18 13:21:49 +08:00
|
|
|
<< PostInit->getSourceRange()
|
|
|
|
<< PreNarrowingType.getLocalUnqualifiedType()
|
|
|
|
<< EntityType.getLocalUnqualifiedType();
|
|
|
|
break;
|
|
|
|
|
|
|
|
case NK_Constant_Narrowing:
|
|
|
|
// A constant value was narrowed.
|
|
|
|
S.Diag(PostInit->getLocStart(),
|
2013-11-12 10:41:45 +08:00
|
|
|
(S.getLangOpts().MicrosoftExt || !S.getLangOpts().CPlusPlus11)
|
|
|
|
? diag::warn_init_list_constant_narrowing
|
|
|
|
: diag::ext_init_list_constant_narrowing)
|
2012-01-18 13:21:49 +08:00
|
|
|
<< PostInit->getSourceRange()
|
2012-03-24 07:55:39 +08:00
|
|
|
<< ConstantValue.getAsString(S.getASTContext(), ConstantType)
|
2011-08-29 23:59:37 +08:00
|
|
|
<< EntityType.getLocalUnqualifiedType();
|
2012-01-18 13:21:49 +08:00
|
|
|
break;
|
|
|
|
|
|
|
|
case NK_Variable_Narrowing:
|
|
|
|
// A variable's value may have been narrowed.
|
|
|
|
S.Diag(PostInit->getLocStart(),
|
2013-11-12 10:41:45 +08:00
|
|
|
(S.getLangOpts().MicrosoftExt || !S.getLangOpts().CPlusPlus11)
|
|
|
|
? diag::warn_init_list_variable_narrowing
|
|
|
|
: diag::ext_init_list_variable_narrowing)
|
2012-01-18 13:21:49 +08:00
|
|
|
<< PostInit->getSourceRange()
|
|
|
|
<< PreNarrowingType.getLocalUnqualifiedType()
|
2011-08-29 23:59:37 +08:00
|
|
|
<< EntityType.getLocalUnqualifiedType();
|
2012-01-18 13:21:49 +08:00
|
|
|
break;
|
|
|
|
}
|
2011-07-27 07:20:30 +08:00
|
|
|
|
2012-02-05 10:13:05 +08:00
|
|
|
SmallString<128> StaticCast;
|
2011-07-27 07:20:30 +08:00
|
|
|
llvm::raw_svector_ostream OS(StaticCast);
|
|
|
|
OS << "static_cast<";
|
|
|
|
if (const TypedefType *TT = EntityType->getAs<TypedefType>()) {
|
|
|
|
// It's important to use the typedef's name if there is one so that the
|
|
|
|
// fixit doesn't break code using types like int64_t.
|
|
|
|
//
|
|
|
|
// FIXME: This will break if the typedef requires qualification. But
|
|
|
|
// getQualifiedNameAsString() includes non-machine-parsable components.
|
2011-10-15 02:45:37 +08:00
|
|
|
OS << *TT->getDecl();
|
2011-07-27 07:20:30 +08:00
|
|
|
} else if (const BuiltinType *BT = EntityType->getAs<BuiltinType>())
|
2012-03-11 15:00:24 +08:00
|
|
|
OS << BT->getName(S.getLangOpts());
|
2011-07-27 07:20:30 +08:00
|
|
|
else {
|
|
|
|
// Oops, we didn't find the actual type of the variable. Don't emit a fixit
|
|
|
|
// with a broken cast.
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
OS << ">(";
|
2012-01-18 13:21:49 +08:00
|
|
|
S.Diag(PostInit->getLocStart(), diag::note_init_list_narrowing_override)
|
|
|
|
<< PostInit->getSourceRange()
|
|
|
|
<< FixItHint::CreateInsertion(PostInit->getLocStart(), OS.str())
|
2011-07-27 07:20:30 +08:00
|
|
|
<< FixItHint::CreateInsertion(
|
2012-01-18 13:21:49 +08:00
|
|
|
S.getPreprocessor().getLocForEndOfToken(PostInit->getLocEnd()), ")");
|
2011-07-27 07:20:30 +08:00
|
|
|
}
|
|
|
|
|
Switch the initialization required by return statements over to the
new InitializationSequence. This fixes some bugs (e.g., PR5808),
changed some diagnostics, and caused more churn than expected. What's
new:
- InitializationSequence now has a "C conversion sequence" category
and step kind, which falls back to
- Changed the diagnostics for returns to always have the result type
of the function first and the type of the expression second.
CheckSingleAssignmentConstraints to peform checking in C.
- Improved ASTs for initialization of return values. The ASTs now
capture all of the temporaries we need to create, but
intentionally do not bind the tempoary that is actually returned,
so that it won't get destroyed twice.
- Make sure to perform an (elidable!) copy of the class object that
is returned from a class.
- Fix copy elision in CodeGen to properly see through the
subexpressions that occur with elidable copies.
- Give "new" its own entity kind; as with return values and thrown
objects, we don't bind the expression so we don't call a
destructor for it.
Note that, with this patch, I've broken returning move-only types in
C++0x. We'll fix it later, when we tackle NRVO.
llvm-svn: 91669
2009-12-18 13:02:21 +08:00
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
// Initialization helper functions
|
|
|
|
//===----------------------------------------------------------------------===//
|
2011-05-13 06:46:29 +08:00
|
|
|
bool
|
|
|
|
Sema::CanPerformCopyInitialization(const InitializedEntity &Entity,
|
|
|
|
ExprResult Init) {
|
|
|
|
if (Init.isInvalid())
|
|
|
|
return false;
|
|
|
|
|
|
|
|
Expr *InitE = Init.get();
|
|
|
|
assert(InitE && "No initialization expression");
|
|
|
|
|
2012-08-01 06:15:04 +08:00
|
|
|
InitializationKind Kind
|
|
|
|
= InitializationKind::CreateCopy(InitE->getLocStart(), SourceLocation());
|
2013-05-03 23:05:50 +08:00
|
|
|
InitializationSequence Seq(*this, Entity, Kind, InitE);
|
2011-06-05 20:23:28 +08:00
|
|
|
return !Seq.Failed();
|
2011-05-13 06:46:29 +08:00
|
|
|
}
|
|
|
|
|
2011-01-27 15:10:08 +08:00
|
|
|
ExprResult
|
Switch the initialization required by return statements over to the
new InitializationSequence. This fixes some bugs (e.g., PR5808),
changed some diagnostics, and caused more churn than expected. What's
new:
- InitializationSequence now has a "C conversion sequence" category
and step kind, which falls back to
- Changed the diagnostics for returns to always have the result type
of the function first and the type of the expression second.
CheckSingleAssignmentConstraints to peform checking in C.
- Improved ASTs for initialization of return values. The ASTs now
capture all of the temporaries we need to create, but
intentionally do not bind the tempoary that is actually returned,
so that it won't get destroyed twice.
- Make sure to perform an (elidable!) copy of the class object that
is returned from a class.
- Fix copy elision in CodeGen to properly see through the
subexpressions that occur with elidable copies.
- Give "new" its own entity kind; as with return values and thrown
objects, we don't bind the expression so we don't call a
destructor for it.
Note that, with this patch, I've broken returning move-only types in
C++0x. We'll fix it later, when we tackle NRVO.
llvm-svn: 91669
2009-12-18 13:02:21 +08:00
|
|
|
Sema::PerformCopyInitialization(const InitializedEntity &Entity,
|
|
|
|
SourceLocation EqualLoc,
|
2011-07-27 07:20:30 +08:00
|
|
|
ExprResult Init,
|
2012-02-25 07:56:31 +08:00
|
|
|
bool TopLevelOfInitList,
|
|
|
|
bool AllowExplicit) {
|
Switch the initialization required by return statements over to the
new InitializationSequence. This fixes some bugs (e.g., PR5808),
changed some diagnostics, and caused more churn than expected. What's
new:
- InitializationSequence now has a "C conversion sequence" category
and step kind, which falls back to
- Changed the diagnostics for returns to always have the result type
of the function first and the type of the expression second.
CheckSingleAssignmentConstraints to peform checking in C.
- Improved ASTs for initialization of return values. The ASTs now
capture all of the temporaries we need to create, but
intentionally do not bind the tempoary that is actually returned,
so that it won't get destroyed twice.
- Make sure to perform an (elidable!) copy of the class object that
is returned from a class.
- Fix copy elision in CodeGen to properly see through the
subexpressions that occur with elidable copies.
- Give "new" its own entity kind; as with return values and thrown
objects, we don't bind the expression so we don't call a
destructor for it.
Note that, with this patch, I've broken returning move-only types in
C++0x. We'll fix it later, when we tackle NRVO.
llvm-svn: 91669
2009-12-18 13:02:21 +08:00
|
|
|
if (Init.isInvalid())
|
|
|
|
return ExprError();
|
|
|
|
|
2010-11-11 11:21:53 +08:00
|
|
|
Expr *InitE = Init.get();
|
Switch the initialization required by return statements over to the
new InitializationSequence. This fixes some bugs (e.g., PR5808),
changed some diagnostics, and caused more churn than expected. What's
new:
- InitializationSequence now has a "C conversion sequence" category
and step kind, which falls back to
- Changed the diagnostics for returns to always have the result type
of the function first and the type of the expression second.
CheckSingleAssignmentConstraints to peform checking in C.
- Improved ASTs for initialization of return values. The ASTs now
capture all of the temporaries we need to create, but
intentionally do not bind the tempoary that is actually returned,
so that it won't get destroyed twice.
- Make sure to perform an (elidable!) copy of the class object that
is returned from a class.
- Fix copy elision in CodeGen to properly see through the
subexpressions that occur with elidable copies.
- Give "new" its own entity kind; as with return values and thrown
objects, we don't bind the expression so we don't call a
destructor for it.
Note that, with this patch, I've broken returning move-only types in
C++0x. We'll fix it later, when we tackle NRVO.
llvm-svn: 91669
2009-12-18 13:02:21 +08:00
|
|
|
assert(InitE && "No initialization expression?");
|
|
|
|
|
|
|
|
if (EqualLoc.isInvalid())
|
|
|
|
EqualLoc = InitE->getLocStart();
|
|
|
|
|
|
|
|
InitializationKind Kind = InitializationKind::CreateCopy(InitE->getLocStart(),
|
2012-02-25 07:56:31 +08:00
|
|
|
EqualLoc,
|
|
|
|
AllowExplicit);
|
2013-09-22 05:19:19 +08:00
|
|
|
InitializationSequence Seq(*this, Entity, Kind, InitE, TopLevelOfInitList);
|
Switch the initialization required by return statements over to the
new InitializationSequence. This fixes some bugs (e.g., PR5808),
changed some diagnostics, and caused more churn than expected. What's
new:
- InitializationSequence now has a "C conversion sequence" category
and step kind, which falls back to
- Changed the diagnostics for returns to always have the result type
of the function first and the type of the expression second.
CheckSingleAssignmentConstraints to peform checking in C.
- Improved ASTs for initialization of return values. The ASTs now
capture all of the temporaries we need to create, but
intentionally do not bind the tempoary that is actually returned,
so that it won't get destroyed twice.
- Make sure to perform an (elidable!) copy of the class object that
is returned from a class.
- Fix copy elision in CodeGen to properly see through the
subexpressions that occur with elidable copies.
- Give "new" its own entity kind; as with return values and thrown
objects, we don't bind the expression so we don't call a
destructor for it.
Note that, with this patch, I've broken returning move-only types in
C++0x. We'll fix it later, when we tackle NRVO.
llvm-svn: 91669
2009-12-18 13:02:21 +08:00
|
|
|
Init.release();
|
2011-07-27 07:20:30 +08:00
|
|
|
|
2013-05-03 23:05:50 +08:00
|
|
|
ExprResult Result = Seq.Perform(*this, Entity, Kind, InitE);
|
2012-01-18 13:21:49 +08:00
|
|
|
|
|
|
|
return Result;
|
Switch the initialization required by return statements over to the
new InitializationSequence. This fixes some bugs (e.g., PR5808),
changed some diagnostics, and caused more churn than expected. What's
new:
- InitializationSequence now has a "C conversion sequence" category
and step kind, which falls back to
- Changed the diagnostics for returns to always have the result type
of the function first and the type of the expression second.
CheckSingleAssignmentConstraints to peform checking in C.
- Improved ASTs for initialization of return values. The ASTs now
capture all of the temporaries we need to create, but
intentionally do not bind the tempoary that is actually returned,
so that it won't get destroyed twice.
- Make sure to perform an (elidable!) copy of the class object that
is returned from a class.
- Fix copy elision in CodeGen to properly see through the
subexpressions that occur with elidable copies.
- Give "new" its own entity kind; as with return values and thrown
objects, we don't bind the expression so we don't call a
destructor for it.
Note that, with this patch, I've broken returning move-only types in
C++0x. We'll fix it later, when we tackle NRVO.
llvm-svn: 91669
2009-12-18 13:02:21 +08:00
|
|
|
}
|