]> granicus.if.org Git - clang/commitdiff
Part three of PR15721: if we have an invalid CXXDefaultInitExpr, don't crash if
authorRichard Smith <richard-llvm@metafoo.co.uk>
Fri, 13 Sep 2013 20:51:45 +0000 (20:51 +0000)
committerRichard Smith <richard-llvm@metafoo.co.uk>
Fri, 13 Sep 2013 20:51:45 +0000 (20:51 +0000)
we try to constant-evaluate it. Patch by Karthik Bhat, test by me.

git-svn-id: https://llvm.org/svn/llvm-project/cfe/trunk@190722 91177308-0d34-0410-b5e6-96231b3b80d8

lib/AST/ExprConstant.cpp
test/SemaCXX/constant-expression-cxx11.cpp

index 218ce8101b5021a7aa1a0749354c4f0dae78d409..0a2cc7b3d6d53a73a8a126caf5c2ba562b595e40 100644 (file)
@@ -3742,8 +3742,12 @@ public:
     { return StmtVisitorTy::Visit(E->getReplacement()); }
   RetTy VisitCXXDefaultArgExpr(const CXXDefaultArgExpr *E)
     { return StmtVisitorTy::Visit(E->getExpr()); }
-  RetTy VisitCXXDefaultInitExpr(const CXXDefaultInitExpr *E)
-    { return StmtVisitorTy::Visit(E->getExpr()); }
+  RetTy VisitCXXDefaultInitExpr(const CXXDefaultInitExpr *E) {
+    // The initializer may not have been parsed yet, or might be erroneous.
+    if (!E->getExpr())
+      return Error(E);
+    return StmtVisitorTy::Visit(E->getExpr());
+  }
   // We cannot create any objects for which cleanups are required, so there is
   // nothing to do here; all cleanups must come from unevaluated subexpressions.
   RetTy VisitExprWithCleanups(const ExprWithCleanups *E)
index 363ca8c7d9d86cf61705ed5be36d01c1632cef41..c7766878583a350366f2767aee08070bbde357bc 100644 (file)
@@ -1776,3 +1776,21 @@ namespace ZeroSizeTypes {
     return &arr[3] - &arr[0]; // expected-note {{subtraction of pointers to type 'int [0]' of zero size}}
   }
 }
+
+namespace BadDefaultInit {
+  template<int N> struct X { static const int n = N; };
+
+  struct A { // expected-note {{subexpression}}
+    int k = X<A().k>::n; // expected-error {{defaulted default constructor of 'A' cannot be used}} expected-error {{not a constant expression}} expected-note {{in call to 'A()'}}
+  };
+
+  // FIXME: The "constexpr constructor must initialize all members" diagnostic
+  // here is bogus (we discard the k(k) initializer because the parameter 'k'
+  // has been marked invalid).
+  struct B { // expected-note 2{{candidate}}
+    constexpr B( // expected-error {{must initialize all members}} expected-note {{candidate}}
+        int k = X<B().k>::n) : // expected-error {{no matching constructor}}
+      k(k) {}
+    int k; // expected-note {{not initialized}}
+  };
+}