On Fri, Mar 6, 2020 at 4:12 AM Richard Smith <rich...@metafoo.co.uk> wrote: > > I implemented a completely different fix in > a95cc77be154433c37a3110ac9af394b7447fcba. Please can you let me know if it > works out?
Stephan reports that it's still breaking compilations (now with an error diagnostic rather than a crash.) Stephan, can you share the link to the error (off-list)? > > The old approach was to treat statement expressions as being value- and > instantiation-dependent if they appear in a dependent context (matching GCC's > apparent behavior). But it turns out that Clang doesn't really know whether > it's in a dependent context! (In particular, Sema::CurContext might be a > template when substituting into that template, might be the enclosing > context, and might be the resulting declaration if it already exists. This > has other implications -- we at least get access checks wrong in some cases > -- but fixing it would be far to invasive to do for Clang 10.) > The new approach is to walk the body of the statement expression and see if > it contains any value- or instantiation-dependent subexpressions, and treat > the statement expression as being correspondingly dependent if so. > > Hans, the new fix will need some work to backport due to ec3060c (it should > be largely mechanical, but perhaps not completely obvious what the mechanical > steps are). Let me know if you'd like me to do the port. > > On Thu, 5 Mar 2020 at 12:20, Richard Smith <rich...@metafoo.co.uk> wrote: >> >> On Thu, 5 Mar 2020 at 06:13, Benjamin Kramer via cfe-commits >> <cfe-commits@lists.llvm.org> wrote: >>> >>> creduce produced this. It's a crash on invalid, but was created from a >>> valid input. >> >> >> Well, "valid" is unclear when using language extensions, but OK. >> >>> >>> $ cat r.ii >>> template <class a> auto b(a) { >>> auto c = [](auto, int) -> decltype(({})) {}; >>> return c(0, 0); >>> } >>> using d = decltype(b(0)); >>> bool f = d ::e; >> >> >> How important is this testcase? The fix you reverted was fixing a >> release-blocker; is this also a release-blocker in your view? >> >>> >>> $ clang r.ii -std=c++17 -w >>> clang-11: clang/lib/AST/Decl.cpp:2343: clang::APValue >>> *clang::VarDecl::evaluateValue(SmallVectorImpl<clang::PartialDiagnosticAt> >>> &) const: Assertion `!Init->isValueDependent()' failed. >>> >>> On Thu, Mar 5, 2020 at 2:18 PM Benjamin Kramer <benny....@gmail.com> wrote: >>> > >>> > It's still crashing clang, reverted this and >>> > f545ede91c9d9f271e7504282cab7bf509607ead in 66addf8e8036. c-reduce is >>> > still chewing on the reproducer. >>> > >>> > On Wed, Mar 4, 2020 at 10:20 PM Richard Smith via cfe-commits >>> > <cfe-commits@lists.llvm.org> wrote: >>> > > >>> > > We found a regression introduced by this patch; fixed in >>> > > f545ede91c9d9f271e7504282cab7bf509607ead. >>> > > >>> > > On Wed, 4 Mar 2020 at 00:30, Hans Wennborg via cfe-commits >>> > > <cfe-commits@lists.llvm.org> wrote: >>> > >> >>> > >> Pushed to 10.x as 3a843031a5ad83a00d2603f623881cb2b2bf719d. Please let >>> > >> me know if you hear about any follow-up issues. >>> > >> >>> > >> Thanks! >>> > >> >>> > >> On Wed, Mar 4, 2020 at 12:28 AM Richard Smith via cfe-commits >>> > >> <cfe-commits@lists.llvm.org> wrote: >>> > >> > >>> > >> > >>> > >> > Author: Richard Smith >>> > >> > Date: 2020-03-03T15:20:40-08:00 >>> > >> > New Revision: bdad0a1b79273733df9acc1be4e992fa5d70ec56 >>> > >> > >>> > >> > URL: >>> > >> > https://github.com/llvm/llvm-project/commit/bdad0a1b79273733df9acc1be4e992fa5d70ec56 >>> > >> > DIFF: >>> > >> > https://github.com/llvm/llvm-project/commit/bdad0a1b79273733df9acc1be4e992fa5d70ec56.diff >>> > >> > >>> > >> > LOG: PR45083: Mark statement expressions as being dependent if they >>> > >> > appear in >>> > >> > dependent contexts. >>> > >> > >>> > >> > We previously assumed they were neither value- nor >>> > >> > instantiation-dependent under any circumstances, which would lead to >>> > >> > crashes and other misbehavior. >>> > >> > >>> > >> > Added: >>> > >> > >>> > >> > >>> > >> > Modified: >>> > >> > clang/include/clang/AST/Expr.h >>> > >> > clang/include/clang/Sema/Sema.h >>> > >> > clang/lib/AST/ASTImporter.cpp >>> > >> > clang/lib/Parse/ParseExpr.cpp >>> > >> > clang/lib/Sema/SemaExpr.cpp >>> > >> > clang/lib/Sema/SemaExprCXX.cpp >>> > >> > clang/lib/Sema/TreeTransform.h >>> > >> > clang/test/SemaTemplate/dependent-expr.cpp >>> > >> > >>> > >> > Removed: >>> > >> > >>> > >> > >>> > >> > >>> > >> > ################################################################################ >>> > >> > diff --git a/clang/include/clang/AST/Expr.h >>> > >> > b/clang/include/clang/AST/Expr.h >>> > >> > index fcdb0b992134..87f9b883486a 100644 >>> > >> > --- a/clang/include/clang/AST/Expr.h >>> > >> > +++ b/clang/include/clang/AST/Expr.h >>> > >> > @@ -3960,14 +3960,14 @@ class StmtExpr : public Expr { >>> > >> > Stmt *SubStmt; >>> > >> > SourceLocation LParenLoc, RParenLoc; >>> > >> > public: >>> > >> > - // FIXME: Does type-dependence need to be computed >>> > >> > diff erently? >>> > >> > - // FIXME: Do we need to compute instantiation >>> > >> > instantiation-dependence for >>> > >> > - // statements? (ugh!) >>> > >> > StmtExpr(CompoundStmt *substmt, QualType T, >>> > >> > - SourceLocation lp, SourceLocation rp) : >>> > >> > + SourceLocation lp, SourceLocation rp, bool >>> > >> > InDependentContext) : >>> > >> > + // Note: we treat a statement-expression in a dependent context >>> > >> > as always >>> > >> > + // being value- and instantiation-dependent. This matches the >>> > >> > behavior of >>> > >> > + // lambda-expressions and GCC. >>> > >> > Expr(StmtExprClass, T, VK_RValue, OK_Ordinary, >>> > >> > - T->isDependentType(), false, false, false), >>> > >> > - SubStmt(substmt), LParenLoc(lp), RParenLoc(rp) { } >>> > >> > + T->isDependentType(), InDependentContext, >>> > >> > InDependentContext, false), >>> > >> > + SubStmt(substmt), LParenLoc(lp), RParenLoc(rp) {} >>> > >> > >>> > >> > /// Build an empty statement expression. >>> > >> > explicit StmtExpr(EmptyShell Empty) : Expr(StmtExprClass, Empty) >>> > >> > { } >>> > >> > >>> > >> > diff --git a/clang/include/clang/Sema/Sema.h >>> > >> > b/clang/include/clang/Sema/Sema.h >>> > >> > index 2304a9718567..5739808753e3 100644 >>> > >> > --- a/clang/include/clang/Sema/Sema.h >>> > >> > +++ b/clang/include/clang/Sema/Sema.h >>> > >> > @@ -4964,7 +4964,7 @@ class Sema final { >>> > >> > LabelDecl *TheDecl); >>> > >> > >>> > >> > void ActOnStartStmtExpr(); >>> > >> > - ExprResult ActOnStmtExpr(SourceLocation LPLoc, Stmt *SubStmt, >>> > >> > + ExprResult ActOnStmtExpr(Scope *S, SourceLocation LPLoc, Stmt >>> > >> > *SubStmt, >>> > >> > SourceLocation RPLoc); // "({..})" >>> > >> > // Handle the final expression in a statement expression. >>> > >> > ExprResult ActOnStmtExprResult(ExprResult E); >>> > >> > >>> > >> > diff --git a/clang/lib/AST/ASTImporter.cpp >>> > >> > b/clang/lib/AST/ASTImporter.cpp >>> > >> > index 52288318337d..0cf00f6ca15b 100644 >>> > >> > --- a/clang/lib/AST/ASTImporter.cpp >>> > >> > +++ b/clang/lib/AST/ASTImporter.cpp >>> > >> > @@ -6631,8 +6631,9 @@ ExpectedStmt >>> > >> > ASTNodeImporter::VisitStmtExpr(StmtExpr *E) { >>> > >> > if (Err) >>> > >> > return std::move(Err); >>> > >> > >>> > >> > - return new (Importer.getToContext()) StmtExpr( >>> > >> > - ToSubStmt, ToType, ToLParenLoc, ToRParenLoc); >>> > >> > + return new (Importer.getToContext()) >>> > >> > + StmtExpr(ToSubStmt, ToType, ToLParenLoc, ToRParenLoc, >>> > >> > + E->isInstantiationDependent()); >>> > >> > } >>> > >> > >>> > >> > ExpectedStmt ASTNodeImporter::VisitUnaryOperator(UnaryOperator *E) { >>> > >> > >>> > >> > diff --git a/clang/lib/Parse/ParseExpr.cpp >>> > >> > b/clang/lib/Parse/ParseExpr.cpp >>> > >> > index 584de6b87d90..b038e6935d87 100644 >>> > >> > --- a/clang/lib/Parse/ParseExpr.cpp >>> > >> > +++ b/clang/lib/Parse/ParseExpr.cpp >>> > >> > @@ -2655,7 +2655,8 @@ Parser::ParseParenExpression(ParenParseOption >>> > >> > &ExprType, bool stopIfCastExpr, >>> > >> > >>> > >> > // If the substmt parsed correctly, build the AST node. >>> > >> > if (!Stmt.isInvalid()) { >>> > >> > - Result = Actions.ActOnStmtExpr(OpenLoc, Stmt.get(), >>> > >> > Tok.getLocation()); >>> > >> > + Result = Actions.ActOnStmtExpr(getCurScope(), OpenLoc, >>> > >> > Stmt.get(), >>> > >> > + Tok.getLocation()); >>> > >> > } else { >>> > >> > Actions.ActOnStmtExprError(); >>> > >> > } >>> > >> > >>> > >> > diff --git a/clang/lib/Sema/SemaExpr.cpp >>> > >> > b/clang/lib/Sema/SemaExpr.cpp >>> > >> > index f50a77a40510..1870e440199d 100644 >>> > >> > --- a/clang/lib/Sema/SemaExpr.cpp >>> > >> > +++ b/clang/lib/Sema/SemaExpr.cpp >>> > >> > @@ -13911,9 +13911,8 @@ void Sema::ActOnStmtExprError() { >>> > >> > PopExpressionEvaluationContext(); >>> > >> > } >>> > >> > >>> > >> > -ExprResult >>> > >> > -Sema::ActOnStmtExpr(SourceLocation LPLoc, Stmt *SubStmt, >>> > >> > - SourceLocation RPLoc) { // "({..})" >>> > >> > +ExprResult Sema::ActOnStmtExpr(Scope *S, SourceLocation LPLoc, Stmt >>> > >> > *SubStmt, >>> > >> > + SourceLocation RPLoc) { // "({..})" >>> > >> > assert(SubStmt && isa<CompoundStmt>(SubStmt) && "Invalid action >>> > >> > invocation!"); >>> > >> > CompoundStmt *Compound = cast<CompoundStmt>(SubStmt); >>> > >> > >>> > >> > @@ -13942,9 +13941,18 @@ Sema::ActOnStmtExpr(SourceLocation LPLoc, >>> > >> > Stmt *SubStmt, >>> > >> > } >>> > >> > } >>> > >> > >>> > >> > + bool IsDependentContext = false; >>> > >> > + if (S) >>> > >> > + IsDependentContext = S->getTemplateParamParent() != nullptr; >>> > >> > + else >>> > >> > + // FIXME: This is not correct when substituting inside a >>> > >> > templated >>> > >> > + // context that isn't a DeclContext (such as a variable >>> > >> > template). >>> > >> > + IsDependentContext = CurContext->isDependentContext(); >>> > >> > + >>> > >> > // FIXME: Check that expression type is complete/non-abstract; >>> > >> > statement >>> > >> > // expressions are not lvalues. >>> > >> > - Expr *ResStmtExpr = new (Context) StmtExpr(Compound, Ty, LPLoc, >>> > >> > RPLoc); >>> > >> > + Expr *ResStmtExpr = >>> > >> > + new (Context) StmtExpr(Compound, Ty, LPLoc, RPLoc, >>> > >> > IsDependentContext); >>> > >> > if (StmtExprMayBindToTemp) >>> > >> > return MaybeBindToTemporary(ResStmtExpr); >>> > >> > return ResStmtExpr; >>> > >> > >>> > >> > diff --git a/clang/lib/Sema/SemaExprCXX.cpp >>> > >> > b/clang/lib/Sema/SemaExprCXX.cpp >>> > >> > index c8c6d7c95a7f..e521ba1d4af1 100644 >>> > >> > --- a/clang/lib/Sema/SemaExprCXX.cpp >>> > >> > +++ b/clang/lib/Sema/SemaExprCXX.cpp >>> > >> > @@ -6955,8 +6955,9 @@ Stmt *Sema::MaybeCreateStmtWithCleanups(Stmt >>> > >> > *SubStmt) { >>> > >> > // a new AsmStmtWithTemporaries. >>> > >> > CompoundStmt *CompStmt = CompoundStmt::Create( >>> > >> > Context, SubStmt, SourceLocation(), SourceLocation()); >>> > >> > - Expr *E = new (Context) StmtExpr(CompStmt, Context.VoidTy, >>> > >> > SourceLocation(), >>> > >> > - SourceLocation()); >>> > >> > + Expr *E = new (Context) >>> > >> > + StmtExpr(CompStmt, Context.VoidTy, SourceLocation(), >>> > >> > SourceLocation(), >>> > >> > + CurContext->isDependentContext()); >>> > >> > return MaybeCreateExprWithCleanups(E); >>> > >> > } >>> > >> > >>> > >> > >>> > >> > diff --git a/clang/lib/Sema/TreeTransform.h >>> > >> > b/clang/lib/Sema/TreeTransform.h >>> > >> > index 002b73c3a1dd..05b41aa53da6 100644 >>> > >> > --- a/clang/lib/Sema/TreeTransform.h >>> > >> > +++ b/clang/lib/Sema/TreeTransform.h >>> > >> > @@ -2549,10 +2549,9 @@ class TreeTransform { >>> > >> > /// >>> > >> > /// By default, performs semantic analysis to build the new >>> > >> > expression. >>> > >> > /// Subclasses may override this routine to provide >>> > >> > diff erent behavior. >>> > >> > - ExprResult RebuildStmtExpr(SourceLocation LParenLoc, >>> > >> > - Stmt *SubStmt, >>> > >> > - SourceLocation RParenLoc) { >>> > >> > - return getSema().ActOnStmtExpr(LParenLoc, SubStmt, RParenLoc); >>> > >> > + ExprResult RebuildStmtExpr(SourceLocation LParenLoc, Stmt >>> > >> > *SubStmt, >>> > >> > + SourceLocation RParenLoc) { >>> > >> > + return getSema().ActOnStmtExpr(nullptr, LParenLoc, SubStmt, >>> > >> > RParenLoc); >>> > >> > } >>> > >> > >>> > >> > /// Build a new __builtin_choose_expr expression. >>> > >> > @@ -11888,6 +11887,8 @@ >>> > >> > TreeTransform<Derived>::TransformLambdaExpr(LambdaExpr *E) { >>> > >> > NewTrailingRequiresClause = getDerived().TransformExpr(TRC); >>> > >> > >>> > >> > // Create the local class that will describe the lambda. >>> > >> > + // FIXME: KnownDependent below is wrong when substituting inside >>> > >> > a templated >>> > >> > + // context that isn't a DeclContext (such as a variable template). >>> > >> > CXXRecordDecl *OldClass = E->getLambdaClass(); >>> > >> > CXXRecordDecl *Class >>> > >> > = getSema().createLambdaClosureType(E->getIntroducerRange(), >>> > >> > >>> > >> > diff --git a/clang/test/SemaTemplate/dependent-expr.cpp >>> > >> > b/clang/test/SemaTemplate/dependent-expr.cpp >>> > >> > index bb1e239c3490..e333ed927b9e 100644 >>> > >> > --- a/clang/test/SemaTemplate/dependent-expr.cpp >>> > >> > +++ b/clang/test/SemaTemplate/dependent-expr.cpp >>> > >> > @@ -1,5 +1,4 @@ >>> > >> > // RUN: %clang_cc1 -fsyntax-only -verify %s >>> > >> > -// expected-no-diagnostics >>> > >> > >>> > >> > // PR5908 >>> > >> > template <typename Iterator> >>> > >> > @@ -108,3 +107,22 @@ namespace PR18152 { >>> > >> > }; >>> > >> > template struct A<0>; >>> > >> > } >>> > >> > + >>> > >> > +template<typename T> void stmt_expr_1() { >>> > >> > + static_assert( ({ false; }), "" ); >>> > >> > +} >>> > >> > +void stmt_expr_2() { >>> > >> > + static_assert( ({ false; }), "" ); // expected-error {{failed}} >>> > >> > +} >>> > >> > + >>> > >> > +namespace PR45083 { >>> > >> > + struct A { bool x; }; >>> > >> > + >>> > >> > + template<typename> struct B : A { >>> > >> > + void f() { >>> > >> > + const int n = ({ if (x) {} 0; }); >>> > >> > + } >>> > >> > + }; >>> > >> > + >>> > >> > + template void B<int>::f(); >>> > >> > +} >>> > >> > >>> > >> > >>> > >> > >>> > >> > _______________________________________________ >>> > >> > cfe-commits mailing list >>> > >> > cfe-commits@lists.llvm.org >>> > >> > https://lists.llvm.org/cgi-bin/mailman/listinfo/cfe-commits >>> > >> _______________________________________________ >>> > >> cfe-commits mailing list >>> > >> cfe-commits@lists.llvm.org >>> > >> https://lists.llvm.org/cgi-bin/mailman/listinfo/cfe-commits >>> > > >>> > > _______________________________________________ >>> > > cfe-commits mailing list >>> > > cfe-commits@lists.llvm.org >>> > > https://lists.llvm.org/cgi-bin/mailman/listinfo/cfe-commits >>> _______________________________________________ >>> cfe-commits mailing list >>> cfe-commits@lists.llvm.org >>> https://lists.llvm.org/cgi-bin/mailman/listinfo/cfe-commits _______________________________________________ cfe-commits mailing list cfe-commits@lists.llvm.org https://lists.llvm.org/cgi-bin/mailman/listinfo/cfe-commits