Fix PR9488: 'auto' type substitution can fail (for instance, if it creates a reference-to-void type). Don't crash if it does.

Also fix an issue where type source information for the resulting type was being lost.

llvm-svn: 127811
This commit is contained in:
Richard Smith 2011-03-17 16:11:59 +00:00
parent 2ef0c69df1
commit 9647d3ca02
6 changed files with 41 additions and 14 deletions

View File

@ -3879,7 +3879,8 @@ public:
FunctionDecl *&Specialization,
sema::TemplateDeductionInfo &Info);
bool DeduceAutoType(QualType AutoType, Expr *Initializer, QualType &Result);
bool DeduceAutoType(TypeSourceInfo *AutoType, Expr *Initializer,
TypeSourceInfo *&Result);
FunctionTemplateDecl *getMoreSpecializedTemplate(FunctionTemplateDecl *FT1,
FunctionTemplateDecl *FT2,

View File

@ -4664,15 +4664,17 @@ void Sema::AddInitializerToDecl(Decl *RealDecl, Expr *Init,
// C++0x [decl.spec.auto]p6. Deduce the type which 'auto' stands in for.
if (TypeMayContainAuto && VDecl->getType()->getContainedAutoType()) {
QualType DeducedType;
if (!DeduceAutoType(VDecl->getType(), Init, DeducedType)) {
TypeSourceInfo *DeducedType = 0;
if (!DeduceAutoType(VDecl->getTypeSourceInfo(), Init, DeducedType))
Diag(VDecl->getLocation(), diag::err_auto_var_deduction_failure)
<< VDecl->getDeclName() << VDecl->getType() << Init->getType()
<< Init->getSourceRange();
if (!DeducedType) {
RealDecl->setInvalidDecl();
return;
}
VDecl->setType(DeducedType);
VDecl->setTypeSourceInfo(DeducedType);
VDecl->setType(DeducedType->getType());
// If this is a redeclaration, check that the type we just deduced matches
// the previously declared type.

View File

@ -6137,15 +6137,17 @@ void Sema::AddCXXDirectInitializerToDecl(Decl *RealDecl,
}
Expr *Init = Exprs.get()[0];
QualType DeducedType;
if (!DeduceAutoType(VDecl->getType(), Init, DeducedType)) {
TypeSourceInfo *DeducedType = 0;
if (!DeduceAutoType(VDecl->getTypeSourceInfo(), Init, DeducedType))
Diag(VDecl->getLocation(), diag::err_auto_var_deduction_failure)
<< VDecl->getDeclName() << VDecl->getType() << Init->getType()
<< Init->getSourceRange();
if (!DeducedType) {
RealDecl->setInvalidDecl();
return;
}
VDecl->setType(DeducedType);
VDecl->setTypeSourceInfo(DeducedType);
VDecl->setType(DeducedType->getType());
// If this is a redeclaration, check that the type we just deduced matches
// the previously declared type.

View File

@ -848,16 +848,18 @@ Sema::BuildCXXNew(SourceLocation StartLoc, bool UseGlobal,
diag::err_auto_new_ctor_multiple_expressions)
<< AllocType << TypeRange);
}
QualType DeducedType;
if (!DeduceAutoType(AllocType, ConstructorArgs.get()[0], DeducedType))
TypeSourceInfo *DeducedType = 0;
if (!DeduceAutoType(AllocTypeInfo, ConstructorArgs.get()[0], DeducedType))
return ExprError(Diag(StartLoc, diag::err_auto_new_deduction_failure)
<< AllocType
<< ConstructorArgs.get()[0]->getType()
<< TypeRange
<< ConstructorArgs.get()[0]->getSourceRange());
if (!DeducedType)
return ExprError();
AllocType = DeducedType;
AllocTypeInfo = Context.getTrivialTypeSourceInfo(AllocType, StartLoc);
AllocTypeInfo = DeducedType;
AllocType = AllocTypeInfo->getType();
}
// Per C++0x [expr.new]p5, the type being constructed may be a

View File

@ -3003,11 +3003,14 @@ namespace {
///
/// \param Result if type deduction was successful, this will be set to the
/// deduced type. This may still contain undeduced autos if the type is
/// dependent.
/// dependent. This will be set to null if deduction succeeded, but auto
/// substitution failed; the appropriate diagnostic will already have been
/// produced in that case.
///
/// \returns true if deduction succeeded, false if it failed.
bool
Sema::DeduceAutoType(QualType Type, Expr *Init, QualType &Result) {
Sema::DeduceAutoType(TypeSourceInfo *Type, Expr *Init,
TypeSourceInfo *&Result) {
if (Init->isTypeDependent()) {
Result = Type;
return true;
@ -3025,8 +3028,10 @@ Sema::DeduceAutoType(QualType Type, Expr *Init, QualType &Result) {
FixedSizeTemplateParameterList<1> TemplateParams(Loc, Loc, &TemplParamPtr,
Loc);
QualType FuncParam =
TypeSourceInfo *FuncParamInfo =
SubstituteAutoTransform(*this, TemplArg).TransformType(Type);
assert(FuncParamInfo && "substituting template parameter for 'auto' failed");
QualType FuncParam = FuncParamInfo->getType();
// Deduce type of TemplParam in Func(Init)
llvm::SmallVector<DeducedTemplateArgument, 1> Deduced;

View File

@ -0,0 +1,15 @@
// RUN: %clang_cc1 -fsyntax-only -verify %s -std=c++0x
void f() {
auto a = f(); // expected-error {{variable has incomplete type 'void'}}
auto &b = f(); // expected-error {{cannot form a reference to 'void'}}
auto *c = f(); // expected-error {{incompatible initializer of type 'void'}}
auto d(f()); // expected-error {{variable has incomplete type 'void'}}
auto &&e(f()); // expected-error {{cannot form a reference to 'void'}}
auto *g(f()); // expected-error {{incompatible initializer of type 'void'}}
(void)new auto(f()); // expected-error {{allocation of incomplete type 'void'}}
(void)new auto&(f()); // expected-error {{cannot form a reference to 'void'}}
(void)new auto*(f()); // expected-error {{incompatible constructor argument of type 'void'}}
}