]> granicus.if.org Git - clang/commitdiff
Fix linkage of type info and vtable for classes without linkage.
authorEli Friedman <eli.friedman@gmail.com>
Fri, 11 Dec 2009 20:48:18 +0000 (20:48 +0000)
committerEli Friedman <eli.friedman@gmail.com>
Fri, 11 Dec 2009 20:48:18 +0000 (20:48 +0000)
git-svn-id: https://llvm.org/svn/llvm-project/cfe/trunk@91152 91177308-0d34-0410-b5e6-96231b3b80d8

lib/CodeGen/CGRTTI.cpp
lib/CodeGen/CGVtable.cpp
test/CodeGenCXX/vtable-linkage.cpp

index a574703935cf390ec2dd814da7a059a25ac6ff50..a15ba4f2157e6dfc94a4f0d92db2ae4b57f6ca75 100644 (file)
@@ -230,7 +230,7 @@ public:
       return llvm::ConstantExpr::getBitCast(GV, Int8PtrTy);
 
     // If we're in an anonymous namespace, then we always want internal linkage.
-    if (RD->isInAnonymousNamespace())
+    if (RD->isInAnonymousNamespace() || !RD->hasLinkage())
       Linkage = llvm::GlobalVariable::InternalLinkage;
     
     bool Hidden = CGM.getDeclVisibilityMode(RD) == LangOptions::Hidden;
@@ -295,7 +295,7 @@ public:
       return DecideExtern(PT->getPointeeType());
     if (const RecordType *RT = Ty->getAs<RecordType>())
       if (const CXXRecordDecl *RD = dyn_cast<CXXRecordDecl>(RT->getDecl()))
-        return !RD->isInAnonymousNamespace();
+        return !RD->isInAnonymousNamespace() && RD->hasLinkage();
     return true;
   }
 
index 752f69c7c297cca7690027f85b3439ff6eedcdfb..868a34144089bf5251dd904a18794d3030eb4b9d 100644 (file)
@@ -1458,7 +1458,7 @@ void CGVtableInfo::MaybeEmitVtable(GlobalDecl GD) {
   }
 
   llvm::GlobalVariable::LinkageTypes Linkage;
-  if (RD->isInAnonymousNamespace())
+  if (RD->isInAnonymousNamespace() || !RD->hasLinkage())
     Linkage = llvm::GlobalVariable::InternalLinkage;
   else if (KeyFunction && !MD->isInlined())
     Linkage = llvm::GlobalVariable::ExternalLinkage;
index 5bc509b65e32f9b82c6aa977764744b496bb744d..f2d914feed9ede1ddc85b985941ce7022a952cc4 100644 (file)
@@ -28,6 +28,8 @@ struct D {
 
 void D::f() { }
 
+static struct : D { } e;
+
 // B has a key function that is not defined in this translation unit so its vtable
 // has external linkage.
 // CHECK: @_ZTV1B = external constant
@@ -43,6 +45,12 @@ void D::f() { }
 // CHECK: @_ZTI1D = constant
 // CHECK: @_ZTV1D = constant
 
+// The anonymous struct for e has no linkage, so the vtable should have
+// internal linkage.
+// CHECK: @"_ZTS3$_0" = internal constant
+// CHECK: @"_ZTI3$_0" = internal constant
+// CHECK: @"_ZTV3$_0" = internal constant
+
 // The A vtable should have internal linkage since it is inside an anonymous 
 // namespace.
 // CHECK: @_ZTSN12_GLOBAL__N_11AE = internal constant