If a method is virtual and the class key function is in another file, emit the method as available_externally.

Fixes PR6747

llvm-svn: 101757
This commit is contained in:
Rafael Espindola
2010-04-19 00:44:22 +00:00
parent 3fef72f0ba
commit 683fe4fc4c
3 changed files with 28 additions and 1 deletions

View File

@@ -295,6 +295,15 @@ public:
CodeGenVTables(CodeGenModule &CGM)
: CGM(CGM) { }
// isKeyFunctionInAnotherTU - True if this record has a key function and it is
// in another translation unit.
static bool isKeyFunctionInAnotherTU(ASTContext &Context,
const CXXRecordDecl *RD) {
assert (RD->isDynamicClass() && "Non dynamic classes have no key.");
const CXXMethodDecl *KeyFunction = Context.getKeyFunction(RD);
return KeyFunction && !KeyFunction->getBody();
}
/// needsVTTParameter - Return whether the given global decl needs a VTT
/// parameter, which it does if it's a base constructor or destructor with
/// virtual bases.

View File

@@ -314,7 +314,14 @@ GetLinkageForFunction(ASTContext &Context, const FunctionDecl *FD,
if (FD->getTemplateSpecializationKind()
== TSK_ExplicitInstantiationDeclaration)
return CodeGenModule::GVA_C99Inline;
if (const CXXMethodDecl *MD = dyn_cast<CXXMethodDecl>(FD)) {
const CXXRecordDecl *RD = MD->getParent();
if (MD->isVirtual() &&
CodeGenVTables::isKeyFunctionInAnotherTU(Context, RD))
return CodeGenModule::GVA_C99Inline;
}
return CodeGenModule::GVA_CXXInline;
}

View File

@@ -0,0 +1,11 @@
// RUN: %clang_cc1 %s -emit-llvm -o - | FileCheck %s
struct foo {
virtual void bar();
// CHECK: define available_externally void @_ZN3foo3bazEv
virtual void baz() {}
};
void zed() {
foo b;
b.baz();
}