]> granicus.if.org Git - clang/commitdiff
Don't crash-on-valid when an inline function is friend of class template
authorDavid Majnemer <david.majnemer@gmail.com>
Fri, 20 Mar 2015 00:02:27 +0000 (00:02 +0000)
committerDavid Majnemer <david.majnemer@gmail.com>
Fri, 20 Mar 2015 00:02:27 +0000 (00:02 +0000)
We assumed that the most recent declaration of an inline function would
also be inline.  However, a more recent declaration can come from a
friend declaration in a class template that is instantiated at the
definition of the function.

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

lib/AST/ASTContext.cpp
lib/AST/Decl.cpp
test/CodeGenCXX/inline-functions.cpp

index 932bc1cb86d2d8bd95281bd8ef6f1e1fcc4184b3..89e1d52c64374c4e72fc2ec5baeb12c89cdaf351 100644 (file)
@@ -7911,7 +7911,7 @@ static GVALinkage basicGVALinkageForFunction(const ASTContext &Context,
   // Functions specified with extern and inline in -fms-compatibility mode
   // forcibly get emitted.  While the body of the function cannot be later
   // replaced, the function definition cannot be discarded.
-  if (FD->getMostRecentDecl()->isMSExternInline())
+  if (FD->isMSExternInline())
     return GVA_StrongODR;
 
   return GVA_DiscardableODR;
index a7e879e04c84a0807661ecd7222b1c47a74afb76..e700cf859973bf2af46b07e788810a5f89e6fac4 100644 (file)
@@ -2734,7 +2734,8 @@ bool FunctionDecl::isMSExternInline() const {
   if (!Context.getLangOpts().MSVCCompat && !hasAttr<DLLExportAttr>())
     return false;
 
-  for (const FunctionDecl *FD = this; FD; FD = FD->getPreviousDecl())
+  for (const FunctionDecl *FD = getMostRecentDecl(); FD;
+       FD = FD->getPreviousDecl())
     if (FD->getStorageClass() == SC_Extern)
       return true;
 
index 622cfa9536f3d2306e7df0471177f3b89730e2ca..20da1f631809ddfb2ca33aff70ca545535f7f435 100644 (file)
@@ -121,3 +121,18 @@ struct TypeWithInlineMethods {
   // CHECK-NOT: _ZN21TypeWithInlineMethods12NonStaticFunEv
   void NonStaticFun() { StaticFun(); }
 };
+
+namespace PR22959 {
+template <typename>
+struct S;
+
+S<int> Foo();
+
+template <typename>
+struct S {
+  friend S<int> Foo();
+};
+
+__attribute__((used)) inline S<int> Foo() { return S<int>(); }
+// CHECK-LABEL: define linkonce_odr void @_ZN7PR229593FooEv(
+}