From 0e027fb32bab35376fb5370742efabd0ddf5cf23 Mon Sep 17 00:00:00 2001 From: Douglas Gregor Date: Sun, 7 Feb 2010 10:31:35 +0000 Subject: [PATCH] Workaround for friend template instantiation crash in PR5848, from Keir Mierle! llvm-svn: 95517 --- clang/lib/Sema/SemaTemplateInstantiateDecl.cpp | 14 +++++++++++--- clang/test/SemaCXX/templated-friend-decl.cpp | 15 +++++++++++++++ 2 files changed, 26 insertions(+), 3 deletions(-) create mode 100644 clang/test/SemaCXX/templated-friend-decl.cpp diff --git a/clang/lib/Sema/SemaTemplateInstantiateDecl.cpp b/clang/lib/Sema/SemaTemplateInstantiateDecl.cpp index 244b5f511b8d..d7820bbc2e66 100644 --- a/clang/lib/Sema/SemaTemplateInstantiateDecl.cpp +++ b/clang/lib/Sema/SemaTemplateInstantiateDecl.cpp @@ -411,9 +411,17 @@ Decl *TemplateDeclInstantiator::VisitFriendDecl(FriendDecl *D) { Decl *NewND; // Hack to make this work almost well pending a rewrite. - if (ND->getDeclContext()->isRecord()) - NewND = SemaRef.FindInstantiatedDecl(ND, TemplateArgs); - else if (D->wasSpecialization()) { + if (ND->getDeclContext()->isRecord()) { + if (!ND->getDeclContext()->isDependentContext()) { + NewND = SemaRef.FindInstantiatedDecl(ND, TemplateArgs); + } else { + // FIXME: Hack to avoid crashing when incorrectly trying to instantiate + // templated friend declarations. This doesn't produce a correct AST; + // however this is sufficient for some AST analysis. The real solution + // must be put in place during the pending rewrite. See PR5848. + return 0; + } + } else if (D->wasSpecialization()) { // Totally egregious hack to work around PR5866 return 0; } else diff --git a/clang/test/SemaCXX/templated-friend-decl.cpp b/clang/test/SemaCXX/templated-friend-decl.cpp new file mode 100644 index 000000000000..c0034cd72f78 --- /dev/null +++ b/clang/test/SemaCXX/templated-friend-decl.cpp @@ -0,0 +1,15 @@ +// RUN: %clang_cc1 %s + +template +struct Foo { + template + struct Bar {}; + + // The templated declaration for class Bar should not be instantiated when + // Foo is. This is to protect against PR5848; for now, this "parses" but + // requires a rewrite of the templated friend code to be properly fixed. + template + friend struct Bar; +}; + +Foo x;