forked from OSchip/llvm-project
Rework the way we track which declarations are "used" during
compilation, and (hopefully) introduce RAII objects for changing the "potentially evaluated" state at all of the necessary places within Sema and Parser. Other changes: - Set the unevaluated/potentially-evaluated context appropriately during template instantiation. - We now recognize three different states while parsing or instantiating expressions: unevaluated, potentially evaluated, and potentially potentially evaluated (for C++'s typeid). - When we're in a potentially potentially-evaluated context, queue up MarkDeclarationReferenced calls in a stack. For C++ typeid expressions that are potentially evaluated, we will play back these MarkDeclarationReferenced calls when we exit the corresponding potentially potentially-evaluated context. - Non-type template arguments are now parsed as constant expressions, so they are not potentially-evaluated. llvm-svn: 73899
This commit is contained in:
parent
8cbbc7944d
commit
0b6a6242ed
|
@ -637,18 +637,50 @@ public:
|
||||||
// Expression Parsing Callbacks.
|
// Expression Parsing Callbacks.
|
||||||
//===--------------------------------------------------------------------===//
|
//===--------------------------------------------------------------------===//
|
||||||
|
|
||||||
/// \brief Notifies the action when the parser is processing an unevaluated
|
/// \brief Describes how the expressions currently being parsed are
|
||||||
/// operand.
|
/// evaluated at run-time, if at all.
|
||||||
|
enum ExpressionEvaluationContext {
|
||||||
|
/// \brief The current expression and its subexpressions occur within an
|
||||||
|
/// unevaluated operand (C++0x [expr]p8), such as a constant expression
|
||||||
|
/// or the subexpression of \c sizeof, where the type or the value of the
|
||||||
|
/// expression may be significant but no code will be generated to evaluate
|
||||||
|
/// the value of the expression at run time.
|
||||||
|
Unevaluated,
|
||||||
|
|
||||||
|
/// \brief The current expression is potentially evaluated at run time,
|
||||||
|
/// which means that code may be generated to evaluate the value of the
|
||||||
|
/// expression at run time.
|
||||||
|
PotentiallyEvaluated,
|
||||||
|
|
||||||
|
/// \brief The current expression may be potentially evaluated or it may
|
||||||
|
/// be unevaluated, but it is impossible to tell from the lexical context.
|
||||||
|
/// This evaluation context is used primary for the operand of the C++
|
||||||
|
/// \c typeid expression, whose argument is potentially evaluated only when
|
||||||
|
/// it is an lvalue of polymorphic class type (C++ [basic.def.odr]p2).
|
||||||
|
PotentiallyPotentiallyEvaluated
|
||||||
|
};
|
||||||
|
|
||||||
|
/// \brief The parser is entering a new expression evaluation context.
|
||||||
///
|
///
|
||||||
/// \param UnevaluatedOperand true to indicate that the parser is processing
|
/// \param NewContext is the new expression evaluation context.
|
||||||
/// an unevaluated operand, or false otherwise.
|
|
||||||
///
|
///
|
||||||
/// \returns whether the the action module was previously in an unevaluated
|
/// \returns the previous expression evaluation context.
|
||||||
/// operand.
|
virtual ExpressionEvaluationContext
|
||||||
virtual bool setUnevaluatedOperand(bool UnevaluatedOperand) {
|
PushExpressionEvaluationContext(ExpressionEvaluationContext NewContext) {
|
||||||
return false;
|
return PotentiallyEvaluated;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// \brief The parser is existing an expression evaluation context.
|
||||||
|
///
|
||||||
|
/// \param OldContext the expression evaluation context that the parser is
|
||||||
|
/// leaving.
|
||||||
|
///
|
||||||
|
/// \param NewContext the expression evaluation context that the parser is
|
||||||
|
/// returning to.
|
||||||
|
virtual void
|
||||||
|
PopExpressionEvaluationContext(ExpressionEvaluationContext OldContext,
|
||||||
|
ExpressionEvaluationContext NewContext) { }
|
||||||
|
|
||||||
// Primary Expressions.
|
// Primary Expressions.
|
||||||
|
|
||||||
/// \brief Retrieve the source range that corresponds to the given
|
/// \brief Retrieve the source range that corresponds to the given
|
||||||
|
@ -1891,6 +1923,29 @@ public:
|
||||||
virtual void print(llvm::raw_ostream &OS) const;
|
virtual void print(llvm::raw_ostream &OS) const;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
/// \brief RAII object that enters a new expression evaluation context.
|
||||||
|
class EnterExpressionEvaluationContext {
|
||||||
|
/// \brief The action object.
|
||||||
|
Action &Actions;
|
||||||
|
|
||||||
|
/// \brief The previous expression evaluation context.
|
||||||
|
Action::ExpressionEvaluationContext PrevContext;
|
||||||
|
|
||||||
|
/// \brief The current expression evaluation context.
|
||||||
|
Action::ExpressionEvaluationContext CurContext;
|
||||||
|
|
||||||
|
public:
|
||||||
|
EnterExpressionEvaluationContext(Action &Actions,
|
||||||
|
Action::ExpressionEvaluationContext NewContext)
|
||||||
|
: Actions(Actions), CurContext(NewContext) {
|
||||||
|
PrevContext = Actions.PushExpressionEvaluationContext(NewContext);
|
||||||
|
}
|
||||||
|
|
||||||
|
~EnterExpressionEvaluationContext() {
|
||||||
|
Actions.PopExpressionEvaluationContext(CurContext, PrevContext);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
} // end namespace clang
|
} // end namespace clang
|
||||||
|
|
||||||
#endif
|
#endif
|
||||||
|
|
|
@ -104,24 +104,6 @@ class Parser {
|
||||||
GreaterThanIsOperator = OldGreaterThanIsOperator;
|
GreaterThanIsOperator = OldGreaterThanIsOperator;
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
/// \brief RAII object that enters an unevaluated operand.
|
|
||||||
class EnterUnevaluatedOperand {
|
|
||||||
/// \brief The action object.
|
|
||||||
Action &Actions;
|
|
||||||
|
|
||||||
/// \brief Whether we were previously within an unevaluated operand.
|
|
||||||
bool PreviouslyInUnevaluatedOperand;
|
|
||||||
|
|
||||||
public:
|
|
||||||
explicit EnterUnevaluatedOperand(Action &Actions) : Actions(Actions) {
|
|
||||||
PreviouslyInUnevaluatedOperand = Actions.setUnevaluatedOperand(true);
|
|
||||||
}
|
|
||||||
|
|
||||||
~EnterUnevaluatedOperand() {
|
|
||||||
Actions.setUnevaluatedOperand(PreviouslyInUnevaluatedOperand);
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
public:
|
public:
|
||||||
Parser(Preprocessor &PP, Action &Actions);
|
Parser(Preprocessor &PP, Action &Actions);
|
||||||
|
|
|
@ -279,7 +279,8 @@ Parser::OwningExprResult Parser::ParseConstantExpression() {
|
||||||
// C++ [basic.def.odr]p2:
|
// C++ [basic.def.odr]p2:
|
||||||
// An expression is potentially evaluated unless it appears where an
|
// An expression is potentially evaluated unless it appears where an
|
||||||
// integral constant expression is required (see 5.19) [...].
|
// integral constant expression is required (see 5.19) [...].
|
||||||
EnterUnevaluatedOperand Unevaluated(Actions);
|
EnterExpressionEvaluationContext Unevaluated(Actions,
|
||||||
|
Action::Unevaluated);
|
||||||
|
|
||||||
OwningExprResult LHS(ParseCastExpression(false));
|
OwningExprResult LHS(ParseCastExpression(false));
|
||||||
if (LHS.isInvalid()) return move(LHS);
|
if (LHS.isInvalid()) return move(LHS);
|
||||||
|
@ -983,7 +984,8 @@ Parser::ParseExprAfterTypeofSizeofAlignof(const Token &OpTok,
|
||||||
//
|
//
|
||||||
// The GNU typeof and alignof extensions also behave as unevaluated
|
// The GNU typeof and alignof extensions also behave as unevaluated
|
||||||
// operands.
|
// operands.
|
||||||
EnterUnevaluatedOperand Unevaluated(Actions);
|
EnterExpressionEvaluationContext Unevaluated(Actions,
|
||||||
|
Action::Unevaluated);
|
||||||
Operand = ParseCastExpression(true/*isUnaryExpression*/);
|
Operand = ParseCastExpression(true/*isUnaryExpression*/);
|
||||||
} else {
|
} else {
|
||||||
// If it starts with a '(', we know that it is either a parenthesized
|
// If it starts with a '(', we know that it is either a parenthesized
|
||||||
|
@ -999,7 +1001,8 @@ Parser::ParseExprAfterTypeofSizeofAlignof(const Token &OpTok,
|
||||||
//
|
//
|
||||||
// The GNU typeof and alignof extensions also behave as unevaluated
|
// The GNU typeof and alignof extensions also behave as unevaluated
|
||||||
// operands.
|
// operands.
|
||||||
EnterUnevaluatedOperand Unevaluated(Actions);
|
EnterExpressionEvaluationContext Unevaluated(Actions,
|
||||||
|
Action::Unevaluated);
|
||||||
Operand = ParseParenExpression(ExprType, true/*stopIfCastExpr*/,
|
Operand = ParseParenExpression(ExprType, true/*stopIfCastExpr*/,
|
||||||
CastTy, RParenLoc);
|
CastTy, RParenLoc);
|
||||||
CastRange = SourceRange(LParenLoc, RParenLoc);
|
CastRange = SourceRange(LParenLoc, RParenLoc);
|
||||||
|
|
|
@ -384,10 +384,9 @@ Parser::OwningExprResult Parser::ParseCXXTypeid() {
|
||||||
//
|
//
|
||||||
// Note that we can't tell whether the expression is an lvalue of a
|
// Note that we can't tell whether the expression is an lvalue of a
|
||||||
// polymorphic class type until after we've parsed the expression, so
|
// polymorphic class type until after we've parsed the expression, so
|
||||||
// we treat the expression as an unevaluated operand and let semantic
|
// we the expression is potentially potentially evaluated.
|
||||||
// analysis cope with case where the expression is not an unevaluated
|
EnterExpressionEvaluationContext Unevaluated(Actions,
|
||||||
// operand.
|
Action::PotentiallyPotentiallyEvaluated);
|
||||||
EnterUnevaluatedOperand Unevaluated(Actions);
|
|
||||||
Result = ParseExpression();
|
Result = ParseExpression();
|
||||||
|
|
||||||
// Match the ')'.
|
// Match the ')'.
|
||||||
|
|
|
@ -750,7 +750,7 @@ void Parser::AnnotateTemplateIdTokenAsType(const CXXScopeSpec *SS) {
|
||||||
/// ParseTemplateArgument - Parse a C++ template argument (C++ [temp.names]).
|
/// ParseTemplateArgument - Parse a C++ template argument (C++ [temp.names]).
|
||||||
///
|
///
|
||||||
/// template-argument: [C++ 14.2]
|
/// template-argument: [C++ 14.2]
|
||||||
/// assignment-expression
|
/// constant-expression
|
||||||
/// type-id
|
/// type-id
|
||||||
/// id-expression
|
/// id-expression
|
||||||
void *Parser::ParseTemplateArgument(bool &ArgIsType) {
|
void *Parser::ParseTemplateArgument(bool &ArgIsType) {
|
||||||
|
@ -768,7 +768,7 @@ void *Parser::ParseTemplateArgument(bool &ArgIsType) {
|
||||||
return TypeArg.get();
|
return TypeArg.get();
|
||||||
}
|
}
|
||||||
|
|
||||||
OwningExprResult ExprArg = ParseAssignmentExpression();
|
OwningExprResult ExprArg = ParseConstantExpression();
|
||||||
if (ExprArg.isInvalid() || !ExprArg.get())
|
if (ExprArg.isInvalid() || !ExprArg.get())
|
||||||
return 0;
|
return 0;
|
||||||
|
|
||||||
|
|
|
@ -182,7 +182,7 @@ Sema::Sema(Preprocessor &pp, ASTContext &ctxt, ASTConsumer &consumer,
|
||||||
Diags(PP.getDiagnostics()), SourceMgr(PP.getSourceManager()),
|
Diags(PP.getDiagnostics()), SourceMgr(PP.getSourceManager()),
|
||||||
ExternalSource(0), CurContext(0), PreDeclaratorDC(0),
|
ExternalSource(0), CurContext(0), PreDeclaratorDC(0),
|
||||||
CurBlock(0), PackContext(0), IdResolver(pp.getLangOptions()),
|
CurBlock(0), PackContext(0), IdResolver(pp.getLangOptions()),
|
||||||
GlobalNewDeleteDeclared(false), InUnevaluatedOperand(false),
|
GlobalNewDeleteDeclared(false), ExprEvalContext(PotentiallyEvaluated),
|
||||||
CompleteTranslationUnit(CompleteTranslationUnit),
|
CompleteTranslationUnit(CompleteTranslationUnit),
|
||||||
NumSFINAEErrors(0), CurrentInstantiationScope(0) {
|
NumSFINAEErrors(0), CurrentInstantiationScope(0) {
|
||||||
|
|
||||||
|
|
|
@ -29,6 +29,7 @@
|
||||||
#include "llvm/ADT/DenseSet.h"
|
#include "llvm/ADT/DenseSet.h"
|
||||||
#include "llvm/ADT/SmallPtrSet.h"
|
#include "llvm/ADT/SmallPtrSet.h"
|
||||||
#include "llvm/ADT/OwningPtr.h"
|
#include "llvm/ADT/OwningPtr.h"
|
||||||
|
#include <list>
|
||||||
#include <string>
|
#include <string>
|
||||||
#include <vector>
|
#include <vector>
|
||||||
|
|
||||||
|
@ -247,11 +248,18 @@ public:
|
||||||
/// have been declared.
|
/// have been declared.
|
||||||
bool GlobalNewDeleteDeclared;
|
bool GlobalNewDeleteDeclared;
|
||||||
|
|
||||||
/// A flag that indicates when we are processing an unevaluated operand
|
/// The current expression evaluation context.
|
||||||
/// (C++0x [expr]). C99 has the same notion of declarations being
|
ExpressionEvaluationContext ExprEvalContext;
|
||||||
/// "used" and C++03 has the notion of "potentially evaluated", but we
|
|
||||||
/// adopt the C++0x terminology since it is most precise.
|
typedef std::vector<std::pair<SourceLocation, Decl *> >
|
||||||
bool InUnevaluatedOperand;
|
PotentiallyReferencedDecls;
|
||||||
|
|
||||||
|
/// A stack of declarations, each element of which is a set of declarations
|
||||||
|
/// that will be marked as referenced if the corresponding potentially
|
||||||
|
/// potentially evaluated expression is potentially evaluated. Each element
|
||||||
|
/// in the stack corresponds to a PotentiallyPotentiallyEvaluated expression
|
||||||
|
/// evaluation context.
|
||||||
|
std::list<PotentiallyReferencedDecls> PotentiallyReferencedDeclStack;
|
||||||
|
|
||||||
/// \brief Whether the code handled by Sema should be considered a
|
/// \brief Whether the code handled by Sema should be considered a
|
||||||
/// complete translation unit or not.
|
/// complete translation unit or not.
|
||||||
|
@ -1334,12 +1342,13 @@ public:
|
||||||
void DiagnoseSentinelCalls(NamedDecl *D, SourceLocation Loc,
|
void DiagnoseSentinelCalls(NamedDecl *D, SourceLocation Loc,
|
||||||
Expr **Args, unsigned NumArgs);
|
Expr **Args, unsigned NumArgs);
|
||||||
|
|
||||||
virtual bool setUnevaluatedOperand(bool UnevaluatedOperand) {
|
virtual ExpressionEvaluationContext
|
||||||
bool Result = InUnevaluatedOperand;
|
PushExpressionEvaluationContext(ExpressionEvaluationContext NewContext);
|
||||||
InUnevaluatedOperand = UnevaluatedOperand;
|
|
||||||
return Result;
|
virtual void
|
||||||
}
|
PopExpressionEvaluationContext(ExpressionEvaluationContext OldContext,
|
||||||
|
ExpressionEvaluationContext NewContext);
|
||||||
|
|
||||||
void MarkDeclarationReferenced(SourceLocation Loc, Decl *D);
|
void MarkDeclarationReferenced(SourceLocation Loc, Decl *D);
|
||||||
|
|
||||||
// Primary Expressions.
|
// Primary Expressions.
|
||||||
|
|
|
@ -5432,6 +5432,35 @@ bool Sema::VerifyIntegerConstantExpression(const Expr *E, llvm::APSInt *Result){
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
Sema::ExpressionEvaluationContext
|
||||||
|
Sema::PushExpressionEvaluationContext(ExpressionEvaluationContext NewContext) {
|
||||||
|
// Introduce a new set of potentially referenced declarations to the stack.
|
||||||
|
if (NewContext == PotentiallyPotentiallyEvaluated)
|
||||||
|
PotentiallyReferencedDeclStack.push_back(PotentiallyReferencedDecls());
|
||||||
|
|
||||||
|
std::swap(ExprEvalContext, NewContext);
|
||||||
|
return NewContext;
|
||||||
|
}
|
||||||
|
|
||||||
|
void
|
||||||
|
Sema::PopExpressionEvaluationContext(ExpressionEvaluationContext OldContext,
|
||||||
|
ExpressionEvaluationContext NewContext) {
|
||||||
|
ExprEvalContext = NewContext;
|
||||||
|
|
||||||
|
if (OldContext == PotentiallyPotentiallyEvaluated) {
|
||||||
|
// Mark any remaining declarations in the current position of the stack
|
||||||
|
// as "referenced". If they were not meant to be referenced, semantic
|
||||||
|
// analysis would have eliminated them (e.g., in ActOnCXXTypeId).
|
||||||
|
PotentiallyReferencedDecls RemainingDecls;
|
||||||
|
RemainingDecls.swap(PotentiallyReferencedDeclStack.back());
|
||||||
|
PotentiallyReferencedDeclStack.pop_back();
|
||||||
|
|
||||||
|
for (PotentiallyReferencedDecls::iterator I = RemainingDecls.begin(),
|
||||||
|
IEnd = RemainingDecls.end();
|
||||||
|
I != IEnd; ++I)
|
||||||
|
MarkDeclarationReferenced(I->first, I->second);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
/// \brief Note that the given declaration was referenced in the source code.
|
/// \brief Note that the given declaration was referenced in the source code.
|
||||||
///
|
///
|
||||||
|
@ -5456,10 +5485,24 @@ void Sema::MarkDeclarationReferenced(SourceLocation Loc, Decl *D) {
|
||||||
if (CurContext->isDependentContext())
|
if (CurContext->isDependentContext())
|
||||||
return;
|
return;
|
||||||
|
|
||||||
// If we are in an unevaluated operand, don't mark any definitions as used.
|
switch (ExprEvalContext) {
|
||||||
if (InUnevaluatedOperand)
|
case Unevaluated:
|
||||||
return;
|
// We are in an expression that is not potentially evaluated; do nothing.
|
||||||
|
return;
|
||||||
|
|
||||||
|
case PotentiallyEvaluated:
|
||||||
|
// We are in a potentially-evaluated expression, so this declaration is
|
||||||
|
// "used"; handle this below.
|
||||||
|
break;
|
||||||
|
|
||||||
|
case PotentiallyPotentiallyEvaluated:
|
||||||
|
// We are in an expression that may be potentially evaluated; queue this
|
||||||
|
// declaration reference until we know whether the expression is
|
||||||
|
// potentially evaluated.
|
||||||
|
PotentiallyReferencedDeclStack.back().push_back(std::make_pair(Loc, D));
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
// Note that this declaration has been used.
|
// Note that this declaration has been used.
|
||||||
if (CXXConstructorDecl *Constructor = dyn_cast<CXXConstructorDecl>(D)) {
|
if (CXXConstructorDecl *Constructor = dyn_cast<CXXConstructorDecl>(D)) {
|
||||||
if (Constructor->isImplicit() && Constructor->isDefaultConstructor()) {
|
if (Constructor->isImplicit() && Constructor->isDefaultConstructor()) {
|
||||||
|
|
|
@ -71,6 +71,31 @@ Sema::ActOnCXXTypeid(SourceLocation OpLoc, SourceLocation LParenLoc,
|
||||||
|
|
||||||
QualType TypeInfoType = Context.getTypeDeclType(TypeInfoRecordDecl);
|
QualType TypeInfoType = Context.getTypeDeclType(TypeInfoRecordDecl);
|
||||||
|
|
||||||
|
if (!isType) {
|
||||||
|
// C++0x [expr.typeid]p3:
|
||||||
|
// When typeid is applied to an expression other than an lvalue of a
|
||||||
|
// polymorphic class type [...] [the] expression is an unevaluated
|
||||||
|
// operand.
|
||||||
|
|
||||||
|
// FIXME: if the type of the expression is a class type, the class
|
||||||
|
// shall be completely defined.
|
||||||
|
bool isUnevaluatedOperand = true;
|
||||||
|
Expr *E = static_cast<Expr *>(TyOrExpr);
|
||||||
|
if (E && !E->isTypeDependent() && E->isLvalue(Context) == Expr::LV_Valid) {
|
||||||
|
QualType T = E->getType();
|
||||||
|
if (const RecordType *RecordT = T->getAsRecordType()) {
|
||||||
|
CXXRecordDecl *RecordD = cast<CXXRecordDecl>(RecordT->getDecl());
|
||||||
|
if (RecordD->isPolymorphic())
|
||||||
|
isUnevaluatedOperand = false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// If this is an unevaluated operand, clear out the set of declaration
|
||||||
|
// references we have been computing.
|
||||||
|
if (isUnevaluatedOperand)
|
||||||
|
PotentiallyReferencedDeclStack.back().clear();
|
||||||
|
}
|
||||||
|
|
||||||
return Owned(new (Context) CXXTypeidExpr(isType, TyOrExpr,
|
return Owned(new (Context) CXXTypeidExpr(isType, TyOrExpr,
|
||||||
TypeInfoType.withConst(),
|
TypeInfoType.withConst(),
|
||||||
SourceRange(OpLoc, RParenLoc)));
|
SourceRange(OpLoc, RParenLoc)));
|
||||||
|
|
|
@ -420,6 +420,7 @@ InstantiateDependentSizedArrayType(const DependentSizedArrayType *T,
|
||||||
}
|
}
|
||||||
|
|
||||||
// Instantiate the size expression
|
// Instantiate the size expression
|
||||||
|
EnterExpressionEvaluationContext Unevaluated(SemaRef, Action::Unevaluated);
|
||||||
Sema::OwningExprResult InstantiatedArraySize =
|
Sema::OwningExprResult InstantiatedArraySize =
|
||||||
SemaRef.InstantiateExpr(ArraySize, TemplateArgs);
|
SemaRef.InstantiateExpr(ArraySize, TemplateArgs);
|
||||||
if (InstantiatedArraySize.isInvalid())
|
if (InstantiatedArraySize.isInvalid())
|
||||||
|
@ -443,6 +444,10 @@ InstantiateDependentSizedExtVectorType(const DependentSizedExtVectorType *T,
|
||||||
return QualType();
|
return QualType();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// The expression in a dependent-sized extended vector type is not
|
||||||
|
// potentially evaluated.
|
||||||
|
EnterExpressionEvaluationContext Unevaluated(SemaRef, Action::Unevaluated);
|
||||||
|
|
||||||
// Instantiate the size expression.
|
// Instantiate the size expression.
|
||||||
const Expr *SizeExpr = T->getSizeExpr();
|
const Expr *SizeExpr = T->getSizeExpr();
|
||||||
Sema::OwningExprResult InstantiatedArraySize =
|
Sema::OwningExprResult InstantiatedArraySize =
|
||||||
|
@ -520,6 +525,9 @@ TemplateTypeInstantiator::InstantiateTypedefType(const TypedefType *T,
|
||||||
QualType
|
QualType
|
||||||
TemplateTypeInstantiator::InstantiateTypeOfExprType(const TypeOfExprType *T,
|
TemplateTypeInstantiator::InstantiateTypeOfExprType(const TypeOfExprType *T,
|
||||||
unsigned Quals) const {
|
unsigned Quals) const {
|
||||||
|
// The expression in a typeof is not potentially evaluated.
|
||||||
|
EnterExpressionEvaluationContext Unevaluated(SemaRef, Action::Unevaluated);
|
||||||
|
|
||||||
Sema::OwningExprResult E
|
Sema::OwningExprResult E
|
||||||
= SemaRef.InstantiateExpr(T->getUnderlyingExpr(), TemplateArgs);
|
= SemaRef.InstantiateExpr(T->getUnderlyingExpr(), TemplateArgs);
|
||||||
if (E.isInvalid())
|
if (E.isInvalid())
|
||||||
|
@ -1175,6 +1183,9 @@ TemplateArgument Sema::Instantiate(TemplateArgument Arg,
|
||||||
return Arg;
|
return Arg;
|
||||||
|
|
||||||
case TemplateArgument::Expression: {
|
case TemplateArgument::Expression: {
|
||||||
|
// Template argument expressions are not potentially evaluated.
|
||||||
|
EnterExpressionEvaluationContext Unevaluated(*this, Action::Unevaluated);
|
||||||
|
|
||||||
Sema::OwningExprResult E = InstantiateExpr(Arg.getAsExpr(), TemplateArgs);
|
Sema::OwningExprResult E = InstantiateExpr(Arg.getAsExpr(), TemplateArgs);
|
||||||
if (E.isInvalid())
|
if (E.isInvalid())
|
||||||
return TemplateArgument();
|
return TemplateArgument();
|
||||||
|
|
|
@ -163,6 +163,9 @@ Decl *TemplateDeclInstantiator::VisitFieldDecl(FieldDecl *D) {
|
||||||
if (Invalid)
|
if (Invalid)
|
||||||
BitWidth = 0;
|
BitWidth = 0;
|
||||||
else if (BitWidth) {
|
else if (BitWidth) {
|
||||||
|
// The bit-width expression is not potentially evaluated.
|
||||||
|
EnterExpressionEvaluationContext Unevaluated(SemaRef, Action::Unevaluated);
|
||||||
|
|
||||||
OwningExprResult InstantiatedBitWidth
|
OwningExprResult InstantiatedBitWidth
|
||||||
= SemaRef.InstantiateExpr(BitWidth, TemplateArgs);
|
= SemaRef.InstantiateExpr(BitWidth, TemplateArgs);
|
||||||
if (InstantiatedBitWidth.isInvalid()) {
|
if (InstantiatedBitWidth.isInvalid()) {
|
||||||
|
@ -192,6 +195,9 @@ Decl *TemplateDeclInstantiator::VisitFieldDecl(FieldDecl *D) {
|
||||||
Decl *TemplateDeclInstantiator::VisitStaticAssertDecl(StaticAssertDecl *D) {
|
Decl *TemplateDeclInstantiator::VisitStaticAssertDecl(StaticAssertDecl *D) {
|
||||||
Expr *AssertExpr = D->getAssertExpr();
|
Expr *AssertExpr = D->getAssertExpr();
|
||||||
|
|
||||||
|
// The expression in a static assertion is not potentially evaluated.
|
||||||
|
EnterExpressionEvaluationContext Unevaluated(SemaRef, Action::Unevaluated);
|
||||||
|
|
||||||
OwningExprResult InstantiatedAssertExpr
|
OwningExprResult InstantiatedAssertExpr
|
||||||
= SemaRef.InstantiateExpr(AssertExpr, TemplateArgs);
|
= SemaRef.InstantiateExpr(AssertExpr, TemplateArgs);
|
||||||
if (InstantiatedAssertExpr.isInvalid())
|
if (InstantiatedAssertExpr.isInvalid())
|
||||||
|
@ -222,8 +228,13 @@ Decl *TemplateDeclInstantiator::VisitEnumDecl(EnumDecl *D) {
|
||||||
EC != ECEnd; ++EC) {
|
EC != ECEnd; ++EC) {
|
||||||
// The specified value for the enumerator.
|
// The specified value for the enumerator.
|
||||||
OwningExprResult Value = SemaRef.Owned((Expr *)0);
|
OwningExprResult Value = SemaRef.Owned((Expr *)0);
|
||||||
if (Expr *UninstValue = EC->getInitExpr())
|
if (Expr *UninstValue = EC->getInitExpr()) {
|
||||||
|
// The enumerator's value expression is not potentially evaluated.
|
||||||
|
EnterExpressionEvaluationContext Unevaluated(SemaRef,
|
||||||
|
Action::Unevaluated);
|
||||||
|
|
||||||
Value = SemaRef.InstantiateExpr(UninstValue, TemplateArgs);
|
Value = SemaRef.InstantiateExpr(UninstValue, TemplateArgs);
|
||||||
|
}
|
||||||
|
|
||||||
// Drop the initial value and continue.
|
// Drop the initial value and continue.
|
||||||
bool isInvalid = false;
|
bool isInvalid = false;
|
||||||
|
|
|
@ -714,9 +714,17 @@ TemplateExprInstantiator::VisitSizeOfAlignOfExpr(SizeOfAlignOfExpr *E) {
|
||||||
E->getSourceRange());
|
E->getSourceRange());
|
||||||
}
|
}
|
||||||
|
|
||||||
Sema::OwningExprResult Arg = Visit(E->getArgumentExpr());
|
Sema::OwningExprResult Arg(SemaRef);
|
||||||
if (Arg.isInvalid())
|
{
|
||||||
return SemaRef.ExprError();
|
// C++0x [expr.sizeof]p1:
|
||||||
|
// The operand is either an expression, which is an unevaluated operand
|
||||||
|
// [...]
|
||||||
|
EnterExpressionEvaluationContext Unevaluated(SemaRef, Action::Unevaluated);
|
||||||
|
|
||||||
|
Arg = Visit(E->getArgumentExpr());
|
||||||
|
if (Arg.isInvalid())
|
||||||
|
return SemaRef.ExprError();
|
||||||
|
}
|
||||||
|
|
||||||
Sema::OwningExprResult Result
|
Sema::OwningExprResult Result
|
||||||
= SemaRef.CreateSizeOfAlignOfExpr((Expr *)Arg.get(), E->getOperatorLoc(),
|
= SemaRef.CreateSizeOfAlignOfExpr((Expr *)Arg.get(), E->getOperatorLoc(),
|
||||||
|
@ -949,6 +957,12 @@ TemplateExprInstantiator::VisitCXXTypeidExpr(CXXTypeidExpr *E) {
|
||||||
E->getSourceRange().getEnd());
|
E->getSourceRange().getEnd());
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// We don't know whether the expression is potentially evaluated until
|
||||||
|
// after we perform semantic analysis, so the expression is potentially
|
||||||
|
// potentially evaluated.
|
||||||
|
EnterExpressionEvaluationContext Unevaluated(SemaRef,
|
||||||
|
Action::PotentiallyPotentiallyEvaluated);
|
||||||
|
|
||||||
OwningExprResult Operand = Visit(E->getExprOperand());
|
OwningExprResult Operand = Visit(E->getExprOperand());
|
||||||
if (Operand.isInvalid())
|
if (Operand.isInvalid())
|
||||||
return SemaRef.ExprError();
|
return SemaRef.ExprError();
|
||||||
|
|
|
@ -145,6 +145,9 @@ TemplateStmtInstantiator::VisitSwitchCase(SwitchCase *S) {
|
||||||
}
|
}
|
||||||
|
|
||||||
Sema::OwningStmtResult TemplateStmtInstantiator::VisitCaseStmt(CaseStmt *S) {
|
Sema::OwningStmtResult TemplateStmtInstantiator::VisitCaseStmt(CaseStmt *S) {
|
||||||
|
// The case value expressions are not potentially evaluated.
|
||||||
|
EnterExpressionEvaluationContext Unevaluated(SemaRef, Action::Unevaluated);
|
||||||
|
|
||||||
// Instantiate left-hand case value.
|
// Instantiate left-hand case value.
|
||||||
OwningExprResult LHS = SemaRef.InstantiateExpr(S->getLHS(), TemplateArgs);
|
OwningExprResult LHS = SemaRef.InstantiateExpr(S->getLHS(), TemplateArgs);
|
||||||
if (LHS.isInvalid())
|
if (LHS.isInvalid())
|
||||||
|
|
Loading…
Reference in New Issue