Index: clang/docs/ReleaseNotes.rst =================================================================== --- clang/docs/ReleaseNotes.rst +++ clang/docs/ReleaseNotes.rst @@ -132,6 +132,12 @@ of a base class is not called in the constructor of its derived class. - Clang no longer emits ``-Wmissing-variable-declarations`` for variables declared with the ``register`` storage class. +- Clang now diagnoses wider cases of tautological use of consteval if or + ``std::is_constant_evaluated``. This also suppresses some false positives. + (`#43760: `_) + (`#51567: `_) +- Clang now diagnoses narrowing implicit conversions on variable initializers in immediate + function context and on constexpr variable template initializers. Bug Fixes in This Version ------------------------- Index: clang/include/clang/Basic/DiagnosticASTKinds.td =================================================================== --- clang/include/clang/Basic/DiagnosticASTKinds.td +++ clang/include/clang/Basic/DiagnosticASTKinds.td @@ -413,10 +413,6 @@ def note_unimplemented_constexpr_lambda_feature_ast : Note< "unimplemented constexpr lambda feature: %0 (coming soon!)">; -def warn_is_constant_evaluated_always_true_constexpr : Warning< - "'%0' will always evaluate to 'true' in a manifestly constant-evaluated expression">, - InGroup>; - // inline asm related. let CategoryName = "Inline Assembly Issue" in { def err_asm_invalid_escape : Error< Index: clang/include/clang/Basic/DiagnosticSemaKinds.td =================================================================== --- clang/include/clang/Basic/DiagnosticSemaKinds.td +++ clang/include/clang/Basic/DiagnosticSemaKinds.td @@ -1570,8 +1570,8 @@ "the message in a static assertion must be produced by a " "constant expression">; -def warn_consteval_if_always_true : Warning< - "consteval if is always true in an %select{unevaluated|immediate}0 context">, +def warn_tautological_consteval_if : Warning< + "consteval if is always %select{true|false}0 in this context">, InGroup>; def ext_inline_variable : ExtWarn< @@ -8876,6 +8876,9 @@ def warn_side_effects_typeid : Warning< "expression with side effects will be evaluated despite being used as an " "operand to 'typeid'">, InGroup; +def warn_tautological_is_constant_evaluated : Warning< + "'%select{std::is_constant_evaluated|__builtin_is_constant_evaluated}0' will always evaluate to %select{false|true}1 in this context">, + InGroup>; def warn_unused_result : Warning< "ignoring return value of function declared with %0 attribute">, InGroup; Index: clang/include/clang/Parse/Parser.h =================================================================== --- clang/include/clang/Parse/Parser.h +++ clang/include/clang/Parse/Parser.h @@ -2012,12 +2012,10 @@ //===--------------------------------------------------------------------===// // C++ if/switch/while/for condition expression. struct ForRangeInfo; - Sema::ConditionResult ParseCXXCondition(StmtResult *InitStmt, - SourceLocation Loc, - Sema::ConditionKind CK, - bool MissingOK, - ForRangeInfo *FRI = nullptr, - bool EnterForConditionScope = false); + Sema::ConditionResult ParseCXXCondition( + StmtResult *InitStmt, SourceLocation Loc, Sema::ConditionKind CK, + bool MissingOK, ForRangeInfo *FRI = nullptr, + bool EnterForConditionScope = false, SourceLocation ConstexprLoc = {}); DeclGroupPtrTy ParseAliasDeclarationInInitStatement(DeclaratorContext Context, ParsedAttributes &Attrs); @@ -2115,7 +2113,8 @@ Sema::ConditionResult &CondResult, SourceLocation Loc, Sema::ConditionKind CK, SourceLocation &LParenLoc, - SourceLocation &RParenLoc); + SourceLocation &RParenLoc, + SourceLocation ConstexprLoc = {}); StmtResult ParseIfStatement(SourceLocation *TrailingElseLoc); StmtResult ParseSwitchStatement(SourceLocation *TrailingElseLoc); StmtResult ParseWhileStatement(SourceLocation *TrailingElseLoc); Index: clang/include/clang/Sema/Sema.h =================================================================== --- clang/include/clang/Sema/Sema.h +++ clang/include/clang/Sema/Sema.h @@ -1350,6 +1350,9 @@ bool InDiscardedStatement; bool InImmediateFunctionContext; bool InImmediateEscalatingFunctionContext; + // The immediate occurances of consteval if or std::is_constant_evaluated() + // are tautologically false + bool IsRuntimeEvaluated; bool IsCurrentlyCheckingDefaultArgumentOrInitializer = false; @@ -1379,7 +1382,8 @@ NumCleanupObjects(NumCleanupObjects), NumTypos(0), ManglingContextDecl(ManglingContextDecl), ExprContext(ExprContext), InDiscardedStatement(false), InImmediateFunctionContext(false), - InImmediateEscalatingFunctionContext(false) {} + InImmediateEscalatingFunctionContext(false), + IsRuntimeEvaluated(false) {} bool isUnevaluated() const { return Context == ExpressionEvaluationContext::Unevaluated || @@ -1418,6 +1422,10 @@ /// A stack of expression evaluation contexts. SmallVector ExprEvalContexts; + /// Source location of the start of `constexpr` in constexpr-if + /// used for diagnostics + SourceLocation ConstexprIfLoc; + // Set of failed immediate invocations to avoid double diagnosing. llvm::SmallPtrSet FailedImmediateInvocations; Index: clang/lib/AST/ExprConstant.cpp =================================================================== --- clang/lib/AST/ExprConstant.cpp +++ clang/lib/AST/ExprConstant.cpp @@ -12159,21 +12159,6 @@ } case Builtin::BI__builtin_is_constant_evaluated: { - const auto *Callee = Info.CurrentCall->getCallee(); - if (Info.InConstantContext && !Info.CheckingPotentialConstantExpression && - (Info.CallStackDepth == 1 || - (Info.CallStackDepth == 2 && Callee->isInStdNamespace() && - Callee->getIdentifier() && - Callee->getIdentifier()->isStr("is_constant_evaluated")))) { - // FIXME: Find a better way to avoid duplicated diagnostics. - if (Info.EvalStatus.Diag) - Info.report((Info.CallStackDepth == 1) ? E->getExprLoc() - : Info.CurrentCall->CallLoc, - diag::warn_is_constant_evaluated_always_true_constexpr) - << (Info.CallStackDepth == 1 ? "__builtin_is_constant_evaluated" - : "std::is_constant_evaluated"); - } - return Success(Info.InConstantContext, E); } Index: clang/lib/Parse/ParseDecl.cpp =================================================================== --- clang/lib/Parse/ParseDecl.cpp +++ clang/lib/Parse/ParseDecl.cpp @@ -2472,6 +2472,15 @@ return ParseDeclarationAfterDeclaratorAndAttributes(D, TemplateInfo); } +/// Determine whether the given declaration is a global variable or +/// static data member. +static bool isNonlocalVariable(const Decl *D) { + if (const VarDecl *Var = dyn_cast_or_null(D)) + return Var->hasGlobalStorage(); + + return false; +} + Decl *Parser::ParseDeclarationAfterDeclaratorAndAttributes( Declarator &D, const ParsedTemplateInfo &TemplateInfo, ForRangeInit *FRI) { // RAII type used to track whether we're inside an initializer. @@ -2504,6 +2513,36 @@ ThisDecl = nullptr; } }; + struct EnterInitializerExpressionEvaluationContext { + Sema &S; + bool Entered; + + EnterInitializerExpressionEvaluationContext(Sema &S, Declarator &D, + Decl *ThisDecl) + : S(S), Entered(false) { + if (ThisDecl && S.getLangOpts().CPlusPlus && !ThisDecl->isInvalidDecl()) { + Entered = true; + bool RuntimeEvaluated = S.ExprEvalContexts.back().IsRuntimeEvaluated; + Sema::ExpressionEvaluationContext NewEEC = + S.ExprEvalContexts.back().Context; + if ((D.getDeclSpec().getTypeQualifiers() == DeclSpec::TQ_const || + isNonlocalVariable(ThisDecl)) && + S.ExprEvalContexts.back().IsRuntimeEvaluated) { + RuntimeEvaluated = false; + } + if (D.getDeclSpec().hasConstexprSpecifier()) { + NewEEC = Sema::ExpressionEvaluationContext::ConstantEvaluated; + RuntimeEvaluated = false; + } + S.PushExpressionEvaluationContext(NewEEC, ThisDecl); + S.ExprEvalContexts.back().IsRuntimeEvaluated = RuntimeEvaluated; + } + } + ~EnterInitializerExpressionEvaluationContext() { + if (Entered) + S.PopExpressionEvaluationContext(); + } + }; enum class InitKind { Uninitialized, Equal, CXXDirect, CXXBraced }; InitKind TheInitKind; @@ -2602,6 +2641,7 @@ << getLangOpts().CPlusPlus20; } else { InitializerScopeRAII InitScope(*this, D, ThisDecl); + EnterInitializerExpressionEvaluationContext InitEC(Actions, D, ThisDecl); if (Tok.is(tok::code_completion)) { cutOffParsing(); @@ -2649,6 +2689,7 @@ ExprVector Exprs; InitializerScopeRAII InitScope(*this, D, ThisDecl); + EnterInitializerExpressionEvaluationContext InitEC(Actions, D, ThisDecl); auto ThisVarDecl = dyn_cast_or_null(ThisDecl); auto RunSignatureHelp = [&]() { @@ -2699,6 +2740,7 @@ Diag(Tok, diag::warn_cxx98_compat_generalized_initializer_lists); InitializerScopeRAII InitScope(*this, D, ThisDecl); + EnterInitializerExpressionEvaluationContext InitEC(Actions, D, ThisDecl); PreferredType.enterVariableInit(Tok.getLocation(), ThisDecl); ExprResult Init(ParseBraceInitializer()); Index: clang/lib/Parse/ParseDeclCXX.cpp =================================================================== --- clang/lib/Parse/ParseDeclCXX.cpp +++ clang/lib/Parse/ParseDeclCXX.cpp @@ -3230,9 +3230,14 @@ assert(Tok.isOneOf(tok::equal, tok::l_brace) && "Data member initializer not starting with '=' or '{'"); + bool IsConstexpr = false; + if (const auto *VD = dyn_cast_if_present(D)) + IsConstexpr = VD->isConstexpr(); + EnterExpressionEvaluationContext Context( Actions, - isa_and_present(D) + IsConstexpr ? Sema::ExpressionEvaluationContext::ConstantEvaluated + : isa_and_present(D) ? Sema::ExpressionEvaluationContext::PotentiallyEvaluatedIfUsed : Sema::ExpressionEvaluationContext::PotentiallyEvaluated, D); Index: clang/lib/Parse/ParseExpr.cpp =================================================================== --- clang/lib/Parse/ParseExpr.cpp +++ clang/lib/Parse/ParseExpr.cpp @@ -2091,6 +2091,13 @@ }; if (OpKind == tok::l_paren || !LHS.isInvalid()) { if (Tok.isNot(tok::r_paren)) { + // FIXME: arguments in consteval functions are constant expression + // regardless of the evaluation context of callsite. However, we + // cannot know whether the called function is constevasl before the + // declaration is resolved. + bool IsRuntimeEvaluated = + Actions.ExprEvalContexts.back().IsRuntimeEvaluated; + Actions.ExprEvalContexts.back().IsRuntimeEvaluated = false; if (ParseExpressionList(ArgExprs, [&] { PreferredType.enterFunctionArgument(Tok.getLocation(), RunSignatureHelp); @@ -2107,6 +2114,8 @@ for (auto &E : ArgExprs) Actions.CorrectDelayedTyposInExpr(E); } + Actions.ExprEvalContexts.back().IsRuntimeEvaluated = + IsRuntimeEvaluated; } } Index: clang/lib/Parse/ParseExprCXX.cpp =================================================================== --- clang/lib/Parse/ParseExprCXX.cpp +++ clang/lib/Parse/ParseExprCXX.cpp @@ -2040,7 +2040,8 @@ Sema::ConditionResult Parser::ParseCXXCondition(StmtResult *InitStmt, SourceLocation Loc, Sema::ConditionKind CK, bool MissingOK, - ForRangeInfo *FRI, bool EnterForConditionScope) { + ForRangeInfo *FRI, bool EnterForConditionScope, + SourceLocation ConstexprLoc) { // Helper to ensure we always enter a continue/break scope if requested. struct ForConditionScopeRAII { Scope *S; @@ -2097,9 +2098,28 @@ *InitStmt = Actions.ActOnNullStmt(SemiLoc); return ParseCXXCondition(nullptr, Loc, CK, MissingOK); } + bool InitStmtIsExprStmt = false; + if (InitStmt) { + RevertingTentativeParsingAction PA(*this); + SkipUntil(tok::r_paren, tok::semi, StopBeforeMatch); + InitStmtIsExprStmt = Tok.is(tok::semi); + } - // Parse the expression. - ExprResult Expr = ParseExpression(); // expression + ExprResult Expr; // expression + { + EnterExpressionEvaluationContext Consteval( + Actions, Sema::ExpressionEvaluationContext::ConstantEvaluated, + /*LambdaContextDecl=*/nullptr, + Sema::ExpressionEvaluationContextRecord::EK_Other, + /*ShouldEnter=*/CK == Sema::ConditionKind::ConstexprIf && + !InitStmtIsExprStmt); + SourceLocation OuterConstexprIfLoc = Actions.ConstexprIfLoc; + Actions.ConstexprIfLoc = ConstexprLoc; + + // Parse the expression. + Expr = ParseExpression(); // expression + Actions.ConstexprIfLoc = OuterConstexprIfLoc; + } if (Expr.isInvalid()) return Sema::ConditionError(); @@ -2187,6 +2207,21 @@ if (CopyInitialization) ConsumeToken(); + Sema::ExpressionEvaluationContext NewEEC = + Actions.ExprEvalContexts.back().Context; + bool RuntimeEvaluated = Actions.ExprEvalContexts.back().IsRuntimeEvaluated; + if (DS.getTypeQualifiers() == DeclSpec::TQ_const) + RuntimeEvaluated = false; + + if (CK == Sema::ConditionKind::ConstexprIf || DS.hasConstexprSpecifier()) { + RuntimeEvaluated = false; + NewEEC = Sema::ExpressionEvaluationContext::ConstantEvaluated; + } + EnterExpressionEvaluationContext Initializer(Actions, NewEEC, DeclOut); + Actions.ExprEvalContexts.back().IsRuntimeEvaluated = RuntimeEvaluated; + SourceLocation OuterConstexprIfLoc = Actions.ConstexprIfLoc; + Actions.ConstexprIfLoc = ConstexprLoc; + ExprResult InitExpr = ExprError(); if (getLangOpts().CPlusPlus11 && Tok.is(tok::l_brace)) { Diag(Tok.getLocation(), @@ -2213,6 +2248,7 @@ Actions.ActOnInitializerError(DeclOut); Actions.FinalizeDeclaration(DeclOut); + Actions.ConstexprIfLoc = OuterConstexprIfLoc; return Actions.ActOnConditionVariable(DeclOut, Loc, CK); } Index: clang/lib/Parse/ParseStmt.cpp =================================================================== --- clang/lib/Parse/ParseStmt.cpp +++ clang/lib/Parse/ParseStmt.cpp @@ -1292,18 +1292,17 @@ /// errors in the condition. /// Additionally, it will assign the location of the outer-most '(' and ')', /// to LParenLoc and RParenLoc, respectively. -bool Parser::ParseParenExprOrCondition(StmtResult *InitStmt, - Sema::ConditionResult &Cond, - SourceLocation Loc, - Sema::ConditionKind CK, - SourceLocation &LParenLoc, - SourceLocation &RParenLoc) { +bool Parser::ParseParenExprOrCondition( + StmtResult *InitStmt, Sema::ConditionResult &Cond, SourceLocation Loc, + Sema::ConditionKind CK, SourceLocation &LParenLoc, + SourceLocation &RParenLoc, SourceLocation ConstexprLoc) { BalancedDelimiterTracker T(*this, tok::l_paren); T.consumeOpen(); SourceLocation Start = Tok.getLocation(); if (getLangOpts().CPlusPlus) { - Cond = ParseCXXCondition(InitStmt, Loc, CK, false); + Cond = ParseCXXCondition(InitStmt, Loc, CK, false, nullptr, false, + ConstexprLoc); } else { ExprResult CondExpr = ParseExpression(); @@ -1463,12 +1462,13 @@ bool IsConsteval = false; SourceLocation NotLocation; SourceLocation ConstevalLoc; + SourceLocation ConstexprLoc; if (Tok.is(tok::kw_constexpr)) { Diag(Tok, getLangOpts().CPlusPlus17 ? diag::warn_cxx14_compat_constexpr_if : diag::ext_constexpr_if); IsConstexpr = true; - ConsumeToken(); + ConstexprLoc = ConsumeToken(); } else { if (Tok.is(tok::exclaim)) { NotLocation = ConsumeToken(); @@ -1514,7 +1514,7 @@ if (ParseParenExprOrCondition(&InitStmt, Cond, IfLoc, IsConstexpr ? Sema::ConditionKind::ConstexprIf : Sema::ConditionKind::Boolean, - LParen, RParen)) + LParen, RParen, ConstexprLoc)) return StmtError(); if (IsConstexpr) @@ -1557,11 +1557,16 @@ if (NotLocation.isInvalid() && IsConsteval) { Context = Sema::ExpressionEvaluationContext::ImmediateFunctionContext; ShouldEnter = true; + } else if (NotLocation.isValid() && IsConsteval) { + Context = Actions.ExprEvalContexts.back().Context; + ShouldEnter = true; } EnterExpressionEvaluationContext PotentiallyDiscarded( Actions, Context, nullptr, Sema::ExpressionEvaluationContextRecord::EK_Other, ShouldEnter); + if (NotLocation.isValid() && IsConsteval) + Actions.ExprEvalContexts.back().IsRuntimeEvaluated = true; ThenStmt = ParseStatement(&InnerStatementTrailingElseLoc); } @@ -1602,11 +1607,16 @@ if (NotLocation.isValid() && IsConsteval) { Context = Sema::ExpressionEvaluationContext::ImmediateFunctionContext; ShouldEnter = true; + } else if (NotLocation.isInvalid() && IsConsteval) { + Context = Actions.ExprEvalContexts.back().Context; + ShouldEnter = true; } EnterExpressionEvaluationContext PotentiallyDiscarded( Actions, Context, nullptr, Sema::ExpressionEvaluationContextRecord::EK_Other, ShouldEnter); + if (NotLocation.isInvalid() && IsConsteval) + Actions.ExprEvalContexts.back().IsRuntimeEvaluated = true; ElseStmt = ParseStatement(); if (ElseStmt.isUsable()) Index: clang/lib/Sema/SemaChecking.cpp =================================================================== --- clang/lib/Sema/SemaChecking.cpp +++ clang/lib/Sema/SemaChecking.cpp @@ -14760,7 +14760,8 @@ static void CheckImplicitConversion(Sema &S, Expr *E, QualType T, SourceLocation CC, bool *ICContext = nullptr, - bool IsListInit = false) { + bool IsListInit = false, + bool IsConstexprInit = false) { if (E->isTypeDependent() || E->isValueDependent()) return; const Type *Source = S.Context.getCanonicalType(E->getType()).getTypePtr(); @@ -15069,11 +15070,14 @@ SmallString<32> PrettyTargetValue; TargetFloatValue.toString(PrettyTargetValue, TargetPrecision); - S.DiagRuntimeBehavior( - E->getExprLoc(), E, + PartialDiagnostic PD = S.PDiag(diag::warn_impcast_integer_float_precision_constant) - << PrettySourceValue << PrettyTargetValue << E->getType() << T - << E->getSourceRange() << clang::SourceRange(CC)); + << PrettySourceValue << PrettyTargetValue << E->getType() << T + << E->getSourceRange() << clang::SourceRange(CC); + if (IsConstexprInit) + S.Diag(E->getExprLoc(), PD); + else + S.DiagRuntimeBehavior(E->getExprLoc(), E, PD); } } else { // Otherwise, the implicit conversion may lose precision. @@ -15127,11 +15131,14 @@ std::string PrettySourceValue = toString(Value, 10); std::string PrettyTargetValue = PrettyPrintInRange(Value, TargetRange); - S.DiagRuntimeBehavior( - E->getExprLoc(), E, + PartialDiagnostic PD = S.PDiag(diag::warn_impcast_integer_precision_constant) - << PrettySourceValue << PrettyTargetValue << E->getType() << T - << E->getSourceRange() << SourceRange(CC)); + << PrettySourceValue << PrettyTargetValue << E->getType() << T + << E->getSourceRange() << SourceRange(CC); + if (IsConstexprInit) + S.Diag(E->getExprLoc(), PD); + else + S.DiagRuntimeBehavior(E->getExprLoc(), E, PD); return; } @@ -15173,11 +15180,14 @@ std::string PrettySourceValue = toString(Value, 10); std::string PrettyTargetValue = PrettyPrintInRange(Value, TargetRange); - S.DiagRuntimeBehavior( - E->getExprLoc(), E, + PartialDiagnostic PD = S.PDiag(diag::warn_impcast_integer_precision_constant) - << PrettySourceValue << PrettyTargetValue << E->getType() << T - << E->getSourceRange() << SourceRange(CC)); + << PrettySourceValue << PrettyTargetValue << E->getType() << T + << E->getSourceRange() << SourceRange(CC); + if (IsConstexprInit) + S.Diag(E->getExprLoc(), PD); + else + S.DiagRuntimeBehavior(E->getExprLoc(), E, PD); return; } } @@ -15339,6 +15349,17 @@ if (auto *Src = OVE->getSourceExpr()) SourceExpr = Src; + bool IsConstexprInit = + S.isConstantEvaluated() && + isa_and_present(S.ExprEvalContexts.back().ManglingContextDecl); + // Constant-evaluated initializers are not diagnosed by DiagRuntimeBehavior, + // but narrowings from the evaluated result to the variable type should be + // diagnosed. + if (IsConstexprInit && SourceExpr->getType() != T) { + CheckImplicitConversion(S, SourceExpr, T, CC, nullptr, IsListInit, + /*IsConstexprInit=*/true); + } + if (const auto *UO = dyn_cast(SourceExpr)) if (UO->getOpcode() == UO_Not && UO->getSubExpr()->isKnownToHaveBooleanValue()) @@ -15374,7 +15395,7 @@ // Go ahead and check any implicit conversions we might have skipped. // The non-canonical typecheck is just an optimization; // CheckImplicitConversion will filter out dead implicit conversions. - if (SourceExpr->getType() != T) + if (!IsConstexprInit && SourceExpr->getType() != T) CheckImplicitConversion(S, SourceExpr, T, CC, nullptr, IsListInit); // Now continue drilling into this expression. Index: clang/lib/Sema/SemaDecl.cpp =================================================================== --- clang/lib/Sema/SemaDecl.cpp +++ clang/lib/Sema/SemaDecl.cpp @@ -15356,7 +15356,7 @@ // Do not push if it is a lambda because one is already pushed when building // the lambda in ActOnStartOfLambdaDefinition(). - if (!isLambdaCallOperator(FD)) + if (!isLambdaCallOperator(FD)) { // [expr.const]/p14.1 // An expression or conversion is in an immediate function context if it is // potentially evaluated and either: its innermost enclosing non-block scope @@ -15364,6 +15364,9 @@ PushExpressionEvaluationContext( FD->isConsteval() ? ExpressionEvaluationContext::ImmediateFunctionContext : ExprEvalContexts.back().Context); + if (!FD->isConsteval() && !FD->isConstexpr()) + ExprEvalContexts.back().IsRuntimeEvaluated = true; + } // Each ExpressionEvaluationContextRecord also keeps track of whether the // context is nested in an immediate function context, so smaller contexts Index: clang/lib/Sema/SemaDeclCXX.cpp =================================================================== --- clang/lib/Sema/SemaDeclCXX.cpp +++ clang/lib/Sema/SemaDeclCXX.cpp @@ -18228,15 +18228,6 @@ Diag(D->getLocation(), diag::err_illegal_initializer); } -/// Determine whether the given declaration is a global variable or -/// static data member. -static bool isNonlocalVariable(const Decl *D) { - if (const VarDecl *Var = dyn_cast_or_null(D)) - return Var->hasGlobalStorage(); - - return false; -} - /// Invoked when we are about to parse an initializer for the declaration /// 'Dcl'. /// @@ -18259,9 +18250,6 @@ // If we are parsing the initializer for a static data member, push a // new expression evaluation context that is associated with this static // data member. - if (isNonlocalVariable(D)) - PushExpressionEvaluationContext( - ExpressionEvaluationContext::PotentiallyEvaluated, D); } /// Invoked after we are finished parsing an initializer for the declaration D. @@ -18270,9 +18258,6 @@ if (!D || D->isInvalidDecl()) return; - if (isNonlocalVariable(D)) - PopExpressionEvaluationContext(); - if (S && D->isOutOfLine()) ExitDeclaratorContext(S); } Index: clang/lib/Sema/SemaExpr.cpp =================================================================== --- clang/lib/Sema/SemaExpr.cpp +++ clang/lib/Sema/SemaExpr.cpp @@ -7081,6 +7081,32 @@ << FixItHint::CreateInsertion(DRE->getLocation(), "std::"); } +// Diagnose uses of std::is_constant_evaluated or +// __builtin_is_constant_evaluated in contexts where the result is known at +// compile time. +static void DiagnoseTautologicalCallToIsConstantEvaluated(Sema &S, + CallExpr *CE) { + if (S.inTemplateInstantiation() || CE->getBeginLoc().isMacroID()) + return; + if (const FunctionDecl *FD = CE->getDirectCallee()) { + bool IsBuiltin = + FD->getBuiltinID() == Builtin::BI__builtin_is_constant_evaluated; + SourceLocation ConstexprIfLoc = S.ConstexprIfLoc; + + if ((FD->isInStdNamespace() && + FD->getNameAsString() == "is_constant_evaluated") || + IsBuiltin) { + bool AlwaysTrue = S.ExprEvalContexts.back().isConstantEvaluated() || + S.ExprEvalContexts.back().isUnevaluated(); + bool AlwaysFalse = S.ExprEvalContexts.back().IsRuntimeEvaluated; + if (AlwaysTrue || AlwaysFalse) + S.Diag(CE->getBeginLoc(), diag::warn_tautological_is_constant_evaluated) + << IsBuiltin << AlwaysTrue + << FixItHint::CreateRemoval(ConstexprIfLoc); + } + } +} + ExprResult Sema::ActOnCallExpr(Scope *Scope, Expr *Fn, SourceLocation LParenLoc, MultiExprArg ArgExprs, SourceLocation RParenLoc, Expr *ExecConfig) { @@ -7107,8 +7133,10 @@ ExecConfig); if (LangOpts.CPlusPlus) { CallExpr *CE = dyn_cast(Call.get()); - if (CE) + if (CE) { DiagnosedUnqualifiedCallsToStdFunctions(*this, CE); + DiagnoseTautologicalCallToIsConstantEvaluated(*this, CE); + } } return Call; } @@ -18591,6 +18619,8 @@ } else llvm_unreachable("Couldn't infer lambda error message."); + if (auto *VD = dyn_cast_if_present(Rec.ManglingContextDecl)) + VD->setInvalidDecl(); for (const auto *L : Rec.Lambdas) Diag(L->getBeginLoc(), D); } Index: clang/lib/Sema/SemaLambda.cpp =================================================================== --- clang/lib/Sema/SemaLambda.cpp +++ clang/lib/Sema/SemaLambda.cpp @@ -1424,6 +1424,8 @@ PushExpressionEvaluationContext( LSI->CallOperator->isConsteval() ? ExpressionEvaluationContext::ImmediateFunctionContext + : isConstantEvaluated() + ? ExpressionEvaluationContext::ConstantEvaluated : ExpressionEvaluationContext::PotentiallyEvaluated); ExprEvalContexts.back().InImmediateFunctionContext = LSI->CallOperator->isConsteval(); Index: clang/lib/Sema/SemaStmt.cpp =================================================================== --- clang/lib/Sema/SemaStmt.cpp +++ clang/lib/Sema/SemaStmt.cpp @@ -933,16 +933,14 @@ } if (ConstevalOrNegatedConsteval) { - bool Immediate = ExprEvalContexts.back().Context == - ExpressionEvaluationContext::ImmediateFunctionContext; - if (CurContext->isFunctionOrMethod()) { - const auto *FD = - dyn_cast(Decl::castFromDeclContext(CurContext)); - if (FD && FD->isImmediateFunction()) - Immediate = true; - } - if (isUnevaluatedContext() || Immediate) - Diags.Report(IfLoc, diag::warn_consteval_if_always_true) << Immediate; + bool AlwaysTrue = ExprEvalContexts.back().isConstantEvaluated() || + ExprEvalContexts.back().isUnevaluated(); + bool AlwaysFalse = ExprEvalContexts.back().IsRuntimeEvaluated; + if (AlwaysTrue || AlwaysFalse) + Diags.Report(IfLoc, diag::warn_tautological_consteval_if) + << (AlwaysTrue + ? StatementKind == IfStatementKind::ConstevalNegated + : StatementKind == IfStatementKind::ConstevalNonNegated); } return BuildIfStmt(IfLoc, StatementKind, LParenLoc, InitStmt, Cond, RParenLoc, Index: clang/lib/Sema/SemaTemplateInstantiateDecl.cpp =================================================================== --- clang/lib/Sema/SemaTemplateInstantiateDecl.cpp +++ clang/lib/Sema/SemaTemplateInstantiateDecl.cpp @@ -5372,8 +5372,11 @@ Var->setImplicitlyInline(); if (OldVar->getInit()) { - EnterExpressionEvaluationContext Evaluated( - *this, Sema::ExpressionEvaluationContext::PotentiallyEvaluated, Var); + Sema::ExpressionEvaluationContext InitEvalContext = + Var->isConstexpr() + ? Sema::ExpressionEvaluationContext::ConstantEvaluated + : Sema::ExpressionEvaluationContext::PotentiallyEvaluated; + EnterExpressionEvaluationContext Evaluated(*this, InitEvalContext, Var); // Instantiate the initializer. ExprResult Init; Index: clang/test/AST/Interp/builtins.cpp =================================================================== --- clang/test/AST/Interp/builtins.cpp +++ clang/test/AST/Interp/builtins.cpp @@ -1,5 +1,5 @@ -// RUN: %clang_cc1 -fexperimental-new-constant-interpreter %s -verify -// RUN: %clang_cc1 -fexperimental-new-constant-interpreter %s -S -emit-llvm -o - | FileCheck %s +// RUN: %clang_cc1 -fexperimental-new-constant-interpreter %s -Wno-constant-evaluated -verify +// RUN: %clang_cc1 -fexperimental-new-constant-interpreter %s -Wno-constant-evaluated -S -emit-llvm -o - | FileCheck %s // RUN: %clang_cc1 -verify=ref %s -Wno-constant-evaluated // RUN: %clang_cc1 -verify=ref %s -Wno-constant-evaluated %s -S -emit-llvm -o - | FileCheck %s Index: clang/test/AST/Interp/if.cpp =================================================================== --- clang/test/AST/Interp/if.cpp +++ clang/test/AST/Interp/if.cpp @@ -1,5 +1,5 @@ -// RUN: %clang_cc1 -std=c++23 -fsyntax-only -fexperimental-new-constant-interpreter %s -verify -// RUN: %clang_cc1 -std=c++23 -fsyntax-only %s -verify=ref +// RUN: %clang_cc1 -std=c++23 -fsyntax-only -fexperimental-new-constant-interpreter -Wno-redundant-consteval-if %s -verify +// RUN: %clang_cc1 -std=c++23 -fsyntax-only -Wno-redundant-consteval-if %s -verify=ref // expected-no-diagnostics // ref-no-diagnostics Index: clang/test/AST/Interp/literals.cpp =================================================================== --- clang/test/AST/Interp/literals.cpp +++ clang/test/AST/Interp/literals.cpp @@ -212,15 +212,11 @@ #if __cplusplus >= 202002L /// FIXME: The following code should be accepted. consteval int foo(int n) { // ref-error {{consteval function never produces a constant expression}} - return sizeof(int[n]); // ref-note 3{{not valid in a constant expression}} \ - // expected-note {{not valid in a constant expression}} + return sizeof(int[n]); // ref-note 2 {{not valid in a constant expression}} } - constinit int var = foo(5); // ref-error {{not a constant expression}} \ - // ref-note 2{{in call to}} \ + constinit int var = foo(5); // ref-note {{in call to}} \ // ref-error {{does not have a constant initializer}} \ // ref-note {{required by 'constinit' specifier}} \ - // expected-error {{is not a constant expression}} \ - // expected-note {{in call to}} \ // expected-error {{does not have a constant initializer}} \ // expected-note {{required by 'constinit' specifier}} \ Index: clang/test/CXX/expr/expr.const/p2-0x.cpp =================================================================== --- clang/test/CXX/expr/expr.const/p2-0x.cpp +++ clang/test/CXX/expr/expr.const/p2-0x.cpp @@ -244,8 +244,8 @@ constexpr int n13 = n5 + n5; // expected-error {{constant expression}} expected-note {{value -4294967296 is outside the range of }} constexpr int n14 = n3 - n5; // expected-error {{constant expression}} expected-note {{value 4294967295 is outside the range of }} constexpr int n15 = n5 * n5; // expected-error {{constant expression}} expected-note {{value 4611686018427387904 is outside the range of }} - constexpr signed char c1 = 100 * 2; // ok expected-warning{{changes value}} - constexpr signed char c2 = '\x64' * '\2'; // also ok expected-warning{{changes value}} + constexpr signed char c1 = 100 * 2; // ok expected-warning {{changes value from 200 to -56}} + constexpr signed char c2 = '\x64' * '\2'; // also ok expected-warning {{changes value from 200 to -56}} constexpr long long ll1 = 0x7fffffffffffffff; // ok constexpr long long ll2 = ll1 + 1; // expected-error {{constant}} expected-note {{ 9223372036854775808 }} constexpr long long ll3 = -ll1 - 1; // ok Index: clang/test/CXX/expr/expr.const/p6-2a.cpp =================================================================== --- clang/test/CXX/expr/expr.const/p6-2a.cpp +++ clang/test/CXX/expr/expr.const/p6-2a.cpp @@ -43,12 +43,11 @@ constexpr Temporary t = {3}; // expected-error {{must have constant destruction}} expected-note {{created here}} expected-note {{in call}} namespace P1073R3 { -consteval int f() { return 42; } // expected-note 2 {{declared here}} +consteval int f() { return 42; } // expected-note {{declared here}} consteval auto g() { return f; } consteval int h(int (*p)() = g()) { return p(); } constexpr int r = h(); -constexpr auto e = g(); // expected-error {{call to consteval function 'P1073R3::g' is not a constant expression}} \ - expected-error {{constexpr variable 'e' must be initialized by a constant expression}} \ - expected-note 2 {{pointer to a consteval declaration is not a constant expression}} +constexpr auto e = g(); // expected-error {{constexpr variable 'e' must be initialized by a constant expression}} \ + expected-note {{pointer to a consteval declaration is not a constant expression}} static_assert(r == 42); } // namespace P1073R3 Index: clang/test/CXX/expr/expr.prim/expr.prim.lambda/p3.cpp =================================================================== --- clang/test/CXX/expr/expr.prim/expr.prim.lambda/p3.cpp +++ clang/test/CXX/expr/expr.prim/expr.prim.lambda/p3.cpp @@ -16,4 +16,5 @@ #if __cplusplus < 201703L // expected-error@-2 {{constexpr variable cannot have non-literal type}} // expected-note@-3 {{lambda closure types are non-literal types before C++17}} +// expected-error@-4 {{a lambda expression may not appear inside of a constant expression}} #endif Index: clang/test/CXX/stmt.stmt/stmt.select/stmt.if/p4.cpp =================================================================== --- clang/test/CXX/stmt.stmt/stmt.select/stmt.if/p4.cpp +++ clang/test/CXX/stmt.stmt/stmt.select/stmt.if/p4.cpp @@ -8,14 +8,14 @@ } else (void)0; // expected-error {{expected { after else}} static_assert([] { - if consteval { + if consteval { // expected-warning {{consteval if is always true}} return 0; } return 1; }() == 0); static_assert([] { - if consteval { + if consteval { // expected-warning {{consteval if is always true}} return 0; } else { return 1; @@ -23,7 +23,7 @@ }() == 0); static_assert([] { - if !consteval { + if !consteval { // expected-warning {{consteval if is always false}} return 0; } else { return 1; @@ -31,14 +31,15 @@ }() == 1); static_assert([] { - if not consteval { + if not consteval { // expected-warning {{consteval if is always false}} return 0; } return 1; }() == 1); if consteval [[likely]] { // expected-warning {{attribute 'likely' has no effect when annotating an 'if consteval' statement}}\ - // expected-note 2{{annotating the 'if consteval' statement here}} + // expected-note 2{{annotating the 'if consteval' statement here}} \ + // expected-warning {{consteval if is always false}} } @@ -49,7 +50,8 @@ } void test_consteval_jumps() { - if consteval { // expected-note 4{{jump enters controlled statement of consteval if}} + if consteval { // expected-warning {{consteval if is always false}} \ + // expected-note 4{{jump enters controlled statement of consteval if}} goto a; goto b; // expected-error {{cannot jump from this goto statement to its label}} a:; @@ -65,14 +67,16 @@ void test_consteval_switch() { int x = 42; switch (x) { - if consteval { // expected-note 2{{jump enters controlled statement of consteval if}} + if consteval { // expected-warning {{consteval if is always false}} \ + // expected-note 2{{jump enters controlled statement of consteval if}} case 1:; // expected-error {{cannot jump from switch statement to this case label}} default:; // expected-error {{cannot jump from switch statement to this case label}} } else { } } switch (x) { - if consteval { // expected-note 2{{jump enters controlled statement of consteval if}} + if consteval { // expected-warning {{consteval if is always false}} \ + // expected-note 2{{jump enters controlled statement of consteval if}} } else { case 2:; // expected-error {{cannot jump from switch statement to this case label}} default:; // expected-error {{cannot jump from switch statement to this case label}} @@ -99,32 +103,32 @@ } consteval void warn_in_consteval() { - if consteval { // expected-warning {{consteval if is always true in an immediate context}} - if consteval {} // expected-warning {{consteval if is always true in an immediate context}} + if consteval { // expected-warning {{consteval if is always true in this context}} + if consteval {} // expected-warning {{consteval if is always true in this context}} } } constexpr void warn_in_consteval2() { if consteval { - if consteval {} // expected-warning {{consteval if is always true in an immediate context}} + if consteval {} // expected-warning {{consteval if is always true in this context}} } } auto y = []() consteval { - if consteval { // expected-warning {{consteval if is always true in an immediate context}} - if consteval {} // expected-warning {{consteval if is always true in an immediate context}} + if consteval { // expected-warning {{consteval if is always true in this context}} + if consteval {} // expected-warning {{consteval if is always true in this context}} } }; namespace test_transform { int f(auto n) { - if consteval { + if consteval { // expected-warning {{consteval if is always false}} n.foo; //expected-error {{no member named}} } else { } - if !consteval { + if !consteval { // expected-warning {{consteval if is always true}} n.foo; //expected-error {{no member named}} } else { Index: clang/test/Parser/pragma-fenv_access.c =================================================================== --- clang/test/Parser/pragma-fenv_access.c +++ clang/test/Parser/pragma-fenv_access.c @@ -33,7 +33,7 @@ CONST float fnot_too_big = not_too_big; CONST int too_big = 0x7ffffff0; #if defined(CPP) -//expected-warning@+2{{implicit conversion}} +//expected-warning@+2{{implicit conversion from 'const int' to 'const float' changes value from 2147483632 to 2147483648}} #endif CONST float fbig = too_big; // inexact #if !defined(CPP) Index: clang/test/SemaCXX/constant-conversion.cpp =================================================================== --- clang/test/SemaCXX/constant-conversion.cpp +++ clang/test/SemaCXX/constant-conversion.cpp @@ -1,4 +1,4 @@ -// RUN: %clang_cc1 -fsyntax-only -verify -triple x86_64-apple-darwin %s +// RUN: %clang_cc1 -fsyntax-only -std=c++20 -verify -triple x86_64-apple-darwin %s // This file tests -Wconstant-conversion, a subcategory of -Wconversion // which is on by default. @@ -31,3 +31,59 @@ s.one_bit = 1; // expected-warning {{implicit truncation from 'int' to a one-bit wide bit-field changes value from 1 to -1}} s.one_bit = true; // no-warning } + +namespace Initializers { +constexpr char ok = true ? 0 : 200; +constexpr char a = 200; // expected-warning {{implicit conversion from 'int' to 'const char' changes value from 200 to -56}} +char b = 200; // expected-warning {{implicit conversion from 'int' to 'char' changes value from 200 to -56}} +const char c = 200; // expected-warning {{implicit conversion from 'int' to 'const char' changes value from 200 to -56}} + +void f() { + constexpr char a = 200; // expected-warning {{implicit conversion from 'int' to 'const char' changes value from 200 to -56}} + char b = 200; // expected-warning {{implicit conversion from 'int' to 'char' changes value from 200 to -56}} + const char c = 200; // expected-warning {{implicit conversion from 'int' to 'const char' changes value from 200 to -56}} + static char d = 2 * 100; // expected-warning {{implicit conversion from 'int' to 'char' changes value from 200 to -56}} +} + +constexpr void g() { + constexpr char a = 2 * 100; // expected-warning {{implicit conversion from 'int' to 'const char' changes value from 200 to -56}} + char b = 2 * 100; // expected-warning {{implicit conversion from 'int' to 'char' changes value from 200 to -56}} + const char c = 2 * 100; // expected-warning {{implicit conversion from 'int' to 'const char' changes value from 200 to -56}} +} + +consteval void h() { + char ok = true ? 0 : 200; + constexpr char a = 200; // expected-warning {{implicit conversion from 'int' to 'const char' changes value from 200 to -56}} + char b = 200; // expected-warning {{implicit conversion from 'int' to 'char' changes value from 200 to -56}} + const char c = 200; // expected-warning {{implicit conversion from 'int' to 'const char' changes value from 200 to -56}} +} + +template +int templ() { + constexpr char a = false ? 129 : N; // expected-warning {{implicit conversion from 'int' to 'const char' changes value from 200 to -56}} \ + // expected-warning {{implicit conversion from 'int' to 'const char' changes value from 345 to 89}} + return 3; +} + +void call_templ() { + int ok = templ<127>(); + int l = templ<3>(); + int m = templ<200>(); // expected-note {{in instantiation of}} + int n = templ<345>(); // expected-note {{in instantiation of}} +} + +template +constexpr signed char diff = a > b ? a - b : b - a; // expected-warning{{changes value from 201 to -55}} \ + // expected-warning{{changes value from 199 to -57}} \ + // expected-warning{{changes value from 299 to 43}} \ + // expected-warning{{changes value from 301 to 45}} + +void test_diff() { + char ok1 = diff<201, 100>; + char ok2 = diff<101, 200>; + char s1 = diff<301, 100>; // expected-note {{in instantiation of}} + char s2 = diff<101, 300>; // expected-note {{in instantiation of}} + char w1 = diff<101, 400>; // expected-note {{in instantiation of}} + char w2 = diff<401, 100>; // expected-note {{in instantiation of}} +} +} Index: clang/test/SemaCXX/constant-expression-cxx11.cpp =================================================================== --- clang/test/SemaCXX/constant-expression-cxx11.cpp +++ clang/test/SemaCXX/constant-expression-cxx11.cpp @@ -1961,7 +1961,7 @@ namespace Lifetime { void f() { - constexpr int &n = n; // expected-error {{constant expression}} expected-note {{use of reference outside its lifetime}} expected-warning {{not yet bound to a value}} + constexpr int &n = n; // expected-error {{constant expression}} expected-note {{use of reference outside its lifetime}} constexpr int m = m; // expected-error {{constant expression}} expected-note {{read of object outside its lifetime}} } Index: clang/test/SemaCXX/cxx2a-consteval.cpp =================================================================== --- clang/test/SemaCXX/cxx2a-consteval.cpp +++ clang/test/SemaCXX/cxx2a-consteval.cpp @@ -713,7 +713,7 @@ struct test { consteval int operator[](int i) const { return {}; } consteval const derp * operator->() const { return &d; } - consteval int f() const { return 12; } // expected-note 2{{declared here}} + consteval int f() const { return 12; } // expected-note {{declared here}} }; constexpr test a; @@ -726,8 +726,7 @@ constexpr int t = a[1]; constexpr int u = a.operator->()->b; constexpr int v = a->b; -// FIXME: I believe this case should work, but we currently reject. -constexpr int w = (a.*&test::f)(); // expected-error {{cannot take address of consteval function 'f' outside of an immediate invocation}} +constexpr int w = (a.*&test::f)(); constexpr int x = a.f(); // Show that we reject when not in an immediate context. @@ -1073,18 +1072,17 @@ consteval const char* make_name(const char* name) { return name;} consteval const char* pad(int P) { return "thestring"; } -int bad = 10; // expected-note 6{{declared here}} +int bad = 10; // expected-note 5{{declared here}} tester glob1(make_name("glob1")); tester glob2(make_name("glob2")); constexpr tester cglob(make_name("cglob")); -tester paddedglob(make_name(pad(bad))); // expected-error {{call to consteval function 'GH58207::make_name' is not a constant expression}} \ +tester paddedglob(make_name(pad(bad))); // expected-error {{call to consteval function 'GH58207::tester::tester' is not a constant expression}} \ // expected-note {{read of non-const variable 'bad' is not allowed in a constant expression}} constexpr tester glob3 = { make_name("glob3") }; -constexpr tester glob4 = { make_name(pad(bad)) }; // expected-error {{call to consteval function 'GH58207::make_name' is not a constant expression}} \ - // expected-error {{constexpr variable 'glob4' must be initialized by a constant expression}} \ - // expected-note 2{{read of non-const variable 'bad' is not allowed in a constant expression}} +constexpr tester glob4 = { make_name(pad(bad)) }; // expected-error {{constexpr variable 'glob4' must be initialized by a constant expression}} \ + // expected-note {{read of non-const variable 'bad' is not allowed in a constant expression}} auto V = make_name(pad(3)); auto V1 = make_name(pad(bad)); // expected-error {{call to consteval function 'GH58207::make_name' is not a constant expression}} \ @@ -1094,12 +1092,12 @@ void foo() { static tester loc1(make_name("loc1")); static constexpr tester loc2(make_name("loc2")); - static tester paddedloc(make_name(pad(bad))); // expected-error {{call to consteval function 'GH58207::make_name' is not a constant expression}} \ + static tester paddedloc(make_name(pad(bad))); // expected-error {{call to consteval function 'GH58207::tester::tester' is not a constant expression}} \ // expected-note {{read of non-const variable 'bad' is not allowed in a constant expression}} } void bar() { - static tester paddedloc(make_name(pad(bad))); // expected-error {{call to consteval function 'GH58207::make_name' is not a constant expression}} \ + static tester paddedloc(make_name(pad(bad))); // expected-error {{call to consteval function 'GH58207::tester::tester' is not a constant expression}} \ // expected-note {{read of non-const variable 'bad' is not allowed in a constant expression}} } } Index: clang/test/SemaCXX/cxx2b-consteval-if.cpp =================================================================== --- clang/test/SemaCXX/cxx2b-consteval-if.cpp +++ clang/test/SemaCXX/cxx2b-consteval-if.cpp @@ -18,7 +18,7 @@ constexpr auto i() { if consteval { - if consteval { // expected-warning {{consteval if is always true in an immediate context}} + if consteval { // expected-warning {{consteval if is always true in this context}} return 1; } return 2; Index: clang/test/SemaCXX/cxx2b-consteval-propagate.cpp =================================================================== --- clang/test/SemaCXX/cxx2b-consteval-propagate.cpp +++ clang/test/SemaCXX/cxx2b-consteval-propagate.cpp @@ -209,7 +209,7 @@ SS::SS(){} // expected-note {{in the default initializer of 'x'}} consteval int f2(int x) { - if (!__builtin_is_constant_evaluated()) side_effect(); + if (!__builtin_is_constant_evaluated()) side_effect(); // expected-warning {{'__builtin_is_constant_evaluated' will always evaluate to true}} return x; } struct S2 { Index: clang/test/SemaCXX/fixit-tautological-meta-constant.cpp =================================================================== --- /dev/null +++ clang/test/SemaCXX/fixit-tautological-meta-constant.cpp @@ -0,0 +1,21 @@ +// RUN: %clang_cc1 -std=c++2b -Wno-unused-value -fdiagnostics-parseable-fixits -fsyntax-only %s 2>&1 | FileCheck %s +namespace std { +constexpr inline bool + is_constant_evaluated() noexcept { + if consteval { return true; } else { return false; } + } +} // namespace std + +constexpr void cexpr() { + if constexpr (std::is_constant_evaluated()) {} + // CHECK: fix-it:"{{.*}}":{[[@LINE-1]]:6-[[@LINE-1]]:16}:"" + // CHECK-NOT: fix-it:"{{.*}}":{[[@LINE-2]]:6-[[@LINE-2]]:16}:"" + constexpr int a = std::is_constant_evaluated(); + // CHECK-NOT: fix-it:"{{.*}}":{[[@LINE-1]]:{{.*}}-[[@LINE-1]]:{{.*}}}:"" + + if constexpr (const int ce = __builtin_is_constant_evaluated()) {} + // CHECK: fix-it:"{{.*}}":{[[@LINE-1]]:6-[[@LINE-1]]:16}:"" + // CHECK-NOT: fix-it:"{{.*}}":{[[@LINE-2]]:6-[[@LINE-2]]:16}:"" + constexpr int b = std::is_constant_evaluated(); + // CHECK-NOT: fix-it:"{{.*}}":{[[@LINE-1]]:{{.*}}-[[@LINE-1]]:{{.*}}}:"" +} Index: clang/test/SemaCXX/vartemplate-lambda.cpp =================================================================== --- clang/test/SemaCXX/vartemplate-lambda.cpp +++ clang/test/SemaCXX/vartemplate-lambda.cpp @@ -12,7 +12,7 @@ struct S { template - static constexpr T t = [](int f = T(7)){return f;}(); // expected-error{{constexpr variable 't' must be initialized by a constant expression}} expected-note{{cannot be used in a constant expression}} + static constexpr T t = [](int f = T(7)){return f;}(); // expected-error{{a lambda expression may not appear inside of a constant expression}} }; template @@ -21,7 +21,7 @@ fn1(a); (void)v1; (void)v1; // expected-note{{in instantiation of variable template specialization 'v1' requested here}} - (void)S::t; // expected-note{{in instantiation of static data member 'S::t' requested here}} + (void)S::t; return 0; } Index: clang/test/SemaCXX/warn-constant-evaluated-constexpr.cpp =================================================================== --- clang/test/SemaCXX/warn-constant-evaluated-constexpr.cpp +++ clang/test/SemaCXX/warn-constant-evaluated-constexpr.cpp @@ -7,35 +7,35 @@ } // namespace std constexpr int fn1() { - if constexpr (std::is_constant_evaluated()) // expected-warning {{'std::is_constant_evaluated' will always evaluate to 'true' in a manifestly constant-evaluated expression}} + if constexpr (std::is_constant_evaluated()) // expected-warning {{'std::is_constant_evaluated' will always evaluate to true in this context}} return 0; else return 1; } constexpr int fn2() { - if constexpr (!std::is_constant_evaluated()) // expected-warning {{'std::is_constant_evaluated' will always evaluate to 'true' in a manifestly constant-evaluated expression}} + if constexpr (!std::is_constant_evaluated()) // expected-warning {{'std::is_constant_evaluated' will always evaluate to true in this context}} return 0; else return 1; } constexpr int fn3() { - if constexpr (std::is_constant_evaluated() == false) // expected-warning {{'std::is_constant_evaluated' will always evaluate to 'true' in a manifestly constant-evaluated expression}} + if constexpr (std::is_constant_evaluated() == false) // expected-warning {{'std::is_constant_evaluated' will always evaluate to true in this context}} return 0; else return 1; } constexpr int fn4() { - if constexpr (__builtin_is_constant_evaluated() == true) // expected-warning {{'__builtin_is_constant_evaluated' will always evaluate to 'true' in a manifestly constant-evaluated expression}} + if constexpr (__builtin_is_constant_evaluated() == true) // expected-warning {{'__builtin_is_constant_evaluated' will always evaluate to true in this context}} return 0; else return 1; } constexpr int fn5() { - if constexpr (__builtin_is_constant_evaluated()) // expected-warning {{'__builtin_is_constant_evaluated' will always evaluate to 'true' in a manifestly constant-evaluated expression}} + if constexpr (__builtin_is_constant_evaluated()) // expected-warning {{'__builtin_is_constant_evaluated' will always evaluate to true in this context}} return 0; else return 1; Index: clang/test/SemaCXX/warn-tautological-meta-constant.cpp =================================================================== --- /dev/null +++ clang/test/SemaCXX/warn-tautological-meta-constant.cpp @@ -0,0 +1,258 @@ +// RUN: %clang_cc1 -std=c++2b -Wno-unused-value -fsyntax-only -verify %s + +namespace std { +constexpr inline bool + is_constant_evaluated() noexcept { + if consteval { return true; } else { return false; } + } +} // namespace std + +namespace P1938 { + constexpr int f1() { + if constexpr (!std::is_constant_evaluated() && sizeof(int) == 4) { // expected-warning {{always evaluate to true}} + return 0; + } + if (std::is_constant_evaluated()) { + return 42; + } else { + if constexpr (std::is_constant_evaluated()) { // expected-warning {{always evaluate to true}} + return 0; + } + } + return 7; +} + + +consteval int f2() { + if (std::is_constant_evaluated() && f1()) { // expected-warning {{always evaluate to true}} + return 42; + } + return 7; +} + + +int f3() { + if (std::is_constant_evaluated() && f1()) { // expected-warning {{always evaluate to false}} + return 42; + } + return 7; +} +} + +void non_qual() { + int ff = std::is_constant_evaluated(); // expected-warning {{always evaluate to false}} + const int aa = std::is_constant_evaluated(); + constexpr int tt = std::is_constant_evaluated(); // expected-warning {{always evaluate to true}} + static int bb = std::is_constant_evaluated(); + constexpr int cc = [](){ + if consteval {return 8;} // expected-warning {{always true}} + }(); + auto lamda = []() { + if consteval {return 8;} + else {return 4;} + }; + auto lamda_const = []() consteval { + if consteval {return 8;} // expected-warning {{always true}} + else {return 4;} + }; + if consteval { // expected-warning {{always false}} + int b = std::is_constant_evaluated(); // expected-warning {{always evaluate to true}} + } +} + +constexpr void in_constexpr() { + int aa = std::is_constant_evaluated(); + constexpr int bb = std::is_constant_evaluated(); // expected-warning {{always evaluate to true}} + const int cc = std::is_constant_evaluated(); + if consteval { + int dd = std::is_constant_evaluated(); // expected-warning {{always evaluate to true}} + constexpr int ee = std::is_constant_evaluated(); // expected-warning {{always evaluate to true}} + const int ff = std::is_constant_evaluated(); // expected-warning {{always evaluate to true}} + } else { + int dd = std::is_constant_evaluated(); // expected-warning {{always evaluate to false}} + constexpr int ee = std::is_constant_evaluated(); // expected-warning {{always evaluate to true}} + const int ff = std::is_constant_evaluated(); + const int qq = std::is_constant_evaluated() ? dd : 3; + } + + if consteval { + if consteval {} // expected-warning {{always true}} + if !consteval {} // expected-warning {{always false}} + } else { + if consteval {} // expected-warning {{always false}} + if !consteval {} // expected-warning {{always true}} + } + if !consteval { + if consteval {} // expected-warning {{always false}} + if !consteval {} // expected-warning {{always true}} + } else { + if consteval {} // expected-warning {{always true}} + if !consteval {} // expected-warning {{always false}} + } +} + +consteval void in_consteval() { + int aa = std::is_constant_evaluated(); // expected-warning {{always evaluate to true}} + constexpr int bb = std::is_constant_evaluated(); // expected-warning {{always evaluate to true}} + const int cc = std::is_constant_evaluated(); // expected-warning {{always evaluate to true}} + auto lambda = []() { + int a(std::is_constant_evaluated()); // expected-warning {{always evaluate to true}} + constexpr int b = std::is_constant_evaluated(); // expected-warning {{always evaluate to true}} + const int c = std::is_constant_evaluated(); // expected-warning {{always evaluate to true}} + }; + if !consteval {} // expected-warning {{always false}} +} + +static_assert(std::is_constant_evaluated()); // expected-warning {{always evaluate to true}} +static_assert(__builtin_is_constant_evaluated()); // expected-warning {{always evaluate to true}} + +template +void templ() { + if constexpr(std::is_constant_evaluated()) {} // expected-warning {{always evaluate to true}} + constexpr bool c = std::is_constant_evaluated(); // expected-warning {{always evaluate to true}} + if consteval {} // expected-warning {{always false}} +} + +template <> void templ() { // expected-warning {{always evaluate to true}} + if constexpr(std::is_constant_evaluated()) {} // expected-warning {{always evaluate to true}} + constexpr bool c = std::is_constant_evaluated(); // expected-warning {{always evaluate to true}} + if consteval {} // expected-warning {{always false}} + templ(); +} + +static_assert([] { + if consteval { // expected-warning {{always true}} + return 0; + } else { + return 1; + } + }() == 0); +constexpr bool b = __builtin_is_constant_evaluated(); // expected-warning {{always evaluate to true}} +constexpr bool c = std::is_constant_evaluated(); // expected-warning {{always evaluate to true}} +constinit bool d = std::is_constant_evaluated(); // expected-warning {{always evaluate to true}} +int p = __builtin_is_constant_evaluated(); +const int q = __builtin_is_constant_evaluated(); + +template // expected-warning {{always evaluate to true}} +void vvv() { + return; +} + +template<> void vvv() {} +template<> void vvv() {} + +template concept C = __builtin_is_constant_evaluated();// expected-warning {{always evaluate to true}} + +struct Foo { + static constexpr bool ce = std::is_constant_evaluated(); // expected-warning {{always evaluate to true}} + const static bool nonce = std::is_constant_evaluated(); + bool b = std::is_constant_evaluated(); + + Foo() { + if constexpr(std::is_constant_evaluated()) {} // expected-warning {{always evaluate to true}} + bool aa = std::is_constant_evaluated(); // expected-warning {{always evaluate to false}} + static bool bb = std::is_constant_evaluated(); + constexpr bool cc = std::is_constant_evaluated(); // expected-warning {{always evaluate to true}} + if consteval {} // expected-warning {{always false}} + } + constexpr Foo(int) { + if constexpr(std::is_constant_evaluated()) {} // expected-warning {{always evaluate to true}} + bool aa = std::is_constant_evaluated(); + static bool bb = std::is_constant_evaluated(); + constexpr bool cc = std::is_constant_evaluated(); // expected-warning {{always evaluate to true}} + } + consteval Foo(int *) { + if constexpr(std::is_constant_evaluated()) {} // expected-warning {{always evaluate to true}} + bool aa = std::is_constant_evaluated(); // expected-warning {{always evaluate to true}} + static bool bb = std::is_constant_evaluated(); // expected-warning {{always evaluate to true}} + constexpr bool cc = std::is_constant_evaluated(); // expected-warning {{always evaluate to true}} + } +}; + +namespace condition { +void f() { + if constexpr (int a = __builtin_is_constant_evaluated(); // expected-warning {{always evaluate to false}} + true) {} + if constexpr (const int a = __builtin_is_constant_evaluated(); + true) {} + if constexpr (constexpr int a = __builtin_is_constant_evaluated(); // expected-warning {{always evaluate to true}} + true) {} + if constexpr (;const int b = __builtin_is_constant_evaluated()) {} // expected-warning {{always evaluate to true}} + if constexpr (;constexpr int b = __builtin_is_constant_evaluated()) {} // expected-warning {{always evaluate to true}} + + if (int a = __builtin_is_constant_evaluated(); // expected-warning {{always evaluate to false}} + true) {} + if (const int a = __builtin_is_constant_evaluated(); + true) {} + if (constexpr int a = __builtin_is_constant_evaluated(); // expected-warning {{always evaluate to true}} + true) {} + if (;int b = __builtin_is_constant_evaluated()) {} // expected-warning {{always evaluate to false}} + if (;const int b = __builtin_is_constant_evaluated()) {} + if (;constexpr int b = __builtin_is_constant_evaluated()) {} // expected-warning {{always evaluate to true}} + + if constexpr (__builtin_is_constant_evaluated()) {} // expected-warning {{always evaluate to true}} + if (__builtin_is_constant_evaluated()) {} // expected-warning {{always evaluate to false}} + + if constexpr (__builtin_is_constant_evaluated(); true) {} // expected-warning {{always evaluate to false}} + // False + if constexpr (({__builtin_is_constant_evaluated();2;3;}); true) {} + + if (__builtin_is_constant_evaluated(); true) {} // expected-warning {{always evaluate to false}} + if constexpr (;__builtin_is_constant_evaluated()) {} // expected-warning {{always evaluate to true}} + if (;__builtin_is_constant_evaluated()) {} // expected-warning {{always evaluate to false}} +} + +constexpr void g() { + if constexpr (int a = __builtin_is_constant_evaluated(); + true) {} + if constexpr (const int a = __builtin_is_constant_evaluated(); + true) {} + if constexpr (constexpr int a = __builtin_is_constant_evaluated(); // expected-warning {{always evaluate to true}} + true) {} + if constexpr (;const int b = __builtin_is_constant_evaluated()) {} // expected-warning {{always evaluate to true}} + if constexpr (;constexpr int b = __builtin_is_constant_evaluated()) {} // expected-warning {{always evaluate to true}} + + if (int a = __builtin_is_constant_evaluated(); + true) {} + if (const int a = __builtin_is_constant_evaluated(); + true) {} + if (constexpr int a = __builtin_is_constant_evaluated(); // expected-warning {{always evaluate to true}} + true) {} + if (;int b = __builtin_is_constant_evaluated()) {} + if (;const int b = __builtin_is_constant_evaluated()) {} + if (;constexpr int b = __builtin_is_constant_evaluated()) {} // expected-warning {{always evaluate to true}} + + if constexpr (__builtin_is_constant_evaluated()) {} // expected-warning {{always evaluate to true}} + if (__builtin_is_constant_evaluated()) {} + + if constexpr (__builtin_is_constant_evaluated(); true) {} + if constexpr (({__builtin_is_constant_evaluated();2;3;}); true) {} + + if (__builtin_is_constant_evaluated(); true) {} + if constexpr (;__builtin_is_constant_evaluated()) {} // expected-warning {{always evaluate to true}} + if (;__builtin_is_constant_evaluated()) {} +} +} + +namespace Arguments { + int nonc(int n) { return n;} + constexpr int cexpr(int n) { return n;} + consteval int ceval(int n) { return n; } + void f() { + // FIXME: These are tauologically-false; + int a1 = nonc(__builtin_is_constant_evaluated()); + const int b1 = nonc(__builtin_is_constant_evaluated()); + int a2 = cexpr(__builtin_is_constant_evaluated()); + + // ok + const int b2 = cexpr(__builtin_is_constant_evaluated()); + constexpr int c2 = cexpr(__builtin_is_constant_evaluated()); // expected-warning {{always evaluate to true}} + + // FIXME: These are tautologically-true; + int a3 = ceval(__builtin_is_constant_evaluated()); + const int b3 = ceval(__builtin_is_constant_evaluated()); + + // ok + constexpr int c3 = ceval(__builtin_is_constant_evaluated()); // expected-warning {{always evaluate to true}} + } +} Index: clang/test/SemaTemplate/concepts.cpp =================================================================== --- clang/test/SemaTemplate/concepts.cpp +++ clang/test/SemaTemplate/concepts.cpp @@ -135,21 +135,21 @@ namespace BuiltinIsConstantEvaluated { // Check that we do all satisfaction and diagnostic checks in a constant context. - template concept C = __builtin_is_constant_evaluated(); // expected-warning {{always}} + template concept C = __builtin_is_constant_evaluated(); // expected-warning {{always evaluate to true}} static_assert(C); - template concept D = __builtin_is_constant_evaluated() == true; // expected-warning {{always}} + template concept D = __builtin_is_constant_evaluated() == true; // expected-warning {{always evaluate to true}} static_assert(D); - template concept E = __builtin_is_constant_evaluated() == true && // expected-warning {{always}} + template concept E = __builtin_is_constant_evaluated() == true && // expected-warning {{always evaluate to true}} false; // expected-note {{'false' evaluated to false}} static_assert(E); // expected-error {{failed}} expected-note {{because 'int' does not satisfy 'E'}} - template concept F = __builtin_is_constant_evaluated() == false; // expected-warning {{always}} + template concept F = __builtin_is_constant_evaluated() == false; // expected-warning {{always evaluate to true}} // expected-note@-1 {{'__builtin_is_constant_evaluated() == false' (1 == 0)}} static_assert(F); // expected-error {{failed}} expected-note {{because 'int' does not satisfy 'F'}} - template concept G = __builtin_is_constant_evaluated() && // expected-warning {{always}} + template concept G = __builtin_is_constant_evaluated() && // expected-warning {{always evaluate to true}} false; // expected-note {{'false' evaluated to false}} static_assert(G); // expected-error {{failed}} expected-note {{because 'int' does not satisfy 'G'}} } Index: clang/unittests/Support/TimeProfilerTest.cpp =================================================================== --- clang/unittests/Support/TimeProfilerTest.cpp +++ clang/unittests/Support/TimeProfilerTest.cpp @@ -188,7 +188,6 @@ | EvaluateAsBooleanCondition () | | EvaluateAsRValue () | EvaluateAsInitializer (slow_value) -| EvaluateAsConstantExpr () | EvaluateAsConstantExpr () | EvaluateAsRValue () | EvaluateAsInitializer (slow_init_list) Index: libcxx/include/__type_traits/is_constant_evaluated.h =================================================================== --- libcxx/include/__type_traits/is_constant_evaluated.h +++ libcxx/include/__type_traits/is_constant_evaluated.h @@ -24,7 +24,11 @@ #endif _LIBCPP_HIDE_FROM_ABI inline _LIBCPP_CONSTEXPR bool __libcpp_is_constant_evaluated() _NOEXCEPT { +#ifndef _LIBCPP_CXX03_LANG return __builtin_is_constant_evaluated(); +#else + return false; +#endif } _LIBCPP_END_NAMESPACE_STD Index: libcxx/test/std/algorithms/alg.modifying.operations/alg.copy/ranges.copy.segmented.pass.cpp =================================================================== --- libcxx/test/std/algorithms/alg.modifying.operations/alg.copy/ranges.copy.segmented.pass.cpp +++ libcxx/test/std/algorithms/alg.modifying.operations/alg.copy/ranges.copy.segmented.pass.cpp @@ -93,12 +93,10 @@ } int main(int, char**) { - if (!std::is_constant_evaluated()) { - test_containers, std::deque>(); - test_containers, std::vector>(); - test_containers, std::deque>(); - test_containers, std::vector>(); - } + test_containers, std::deque>(); + test_containers, std::vector>(); + test_containers, std::deque>(); + test_containers, std::vector>(); types::for_each(types::forward_iterator_list{}, [] { test_join_view(); Index: libcxx/test/std/utilities/meta/meta.const.eval/is_constant_evaluated.verify.cpp =================================================================== --- libcxx/test/std/utilities/meta/meta.const.eval/is_constant_evaluated.verify.cpp +++ libcxx/test/std/utilities/meta/meta.const.eval/is_constant_evaluated.verify.cpp @@ -24,7 +24,7 @@ #else // expected-error-re@+1 {{{{(static_assert|static assertion)}} failed}} static_assert(!std::is_constant_evaluated(), ""); - // expected-warning@-1 0-1 {{'std::is_constant_evaluated' will always evaluate to 'true' in a manifestly constant-evaluated expression}} + // expected-warning-re@-1 0-1 {{'std::is_constant_evaluated' will always evaluate to {{('true' in a manifestly constant-evaluated expression|true in this context)}}}} #endif return 0; }