[Sema] Fix crash on BlockExprs in a default member initializers
authorErik Pilkington <erik.pilkington@gmail.com>
Mon, 23 Jul 2018 22:47:37 +0000 (22:47 +0000)
committerErik Pilkington <erik.pilkington@gmail.com>
Mon, 23 Jul 2018 22:47:37 +0000 (22:47 +0000)
Clang would crash when instantiating a BlockDecl that appeared in a
default-member-initializer of a class template. Fix this by deferring the
instantiation until we instantate the BlockExpr.

rdar://41200624

Differential revision: https://reviews.llvm.org/D49688

llvm-svn: 337766

clang/lib/Sema/SemaTemplateInstantiate.cpp
clang/test/SemaCXX/instantiate-blocks.cpp

index 5a4a6bd..3cf5843 100644 (file)
@@ -2083,6 +2083,11 @@ Sema::InstantiateClass(SourceLocation PointOfInstantiation,
     if (Member->getDeclContext() != Pattern)
       continue;
 
+    // BlockDecls can appear in a default-member-initializer. They must be the
+    // child of a BlockExpr, so we only know how to instantiate them from there.
+    if (isa<BlockDecl>(Member))
+      continue;
+
     if (Member->isInvalidDecl()) {
       Instantiation->setInvalidDecl();
       continue;
index bb0f8d8..dbcef50 100644 (file)
@@ -30,3 +30,12 @@ int main(void)
    noret((float)0.0, double(0.0)); // expected-note {{in instantiation of function template specialization 'noret<float, double>' requested here}}
 }
 
+namespace rdar41200624 {
+template <class T>
+struct S {
+  int (^p)() = ^{ return 0; };
+  T (^t)() = ^{ return T{}; };
+  T s = ^{ return T{}; }();
+};
+S<int> x;
+}