Support attribute used in member funcs of class templates

Summary:
As PR17480 describes, clang does not support the used attribute
for member functions of class templates. This means that if the member
function is not used, its definition is never instantiated. This patch
changes clang to emit the definition if it has the used attribute.

Test Plan: Added a testcase

Reviewed By: aaron.ballman

Differential Revision: https://reviews.llvm.org/D56928

llvm-svn: 352740
This commit is contained in:
Rafael Auler 2019-01-31 09:38:31 +00:00
parent f392bc846f
commit 4b70204588
2 changed files with 33 additions and 0 deletions

View File

@ -2175,6 +2175,20 @@ TemplateDeclInstantiator::VisitCXXMethodDecl(CXXMethodDecl *D,
Owner->addDecl(Method);
}
// PR17480: Honor the used attribute to instantiate member function
// definitions
if (Method->hasAttr<UsedAttr>()) {
if (const auto *A = dyn_cast<CXXRecordDecl>(Owner)) {
SourceLocation Loc;
if (const MemberSpecializationInfo *MSInfo =
A->getMemberSpecializationInfo())
Loc = MSInfo->getPointOfInstantiation();
else if (const auto *Spec = dyn_cast<ClassTemplateSpecializationDecl>(A))
Loc = Spec->getPointOfInstantiation();
SemaRef.MarkFunctionReferenced(Loc, Method);
}
}
return Method;
}

View File

@ -0,0 +1,19 @@
// RUN: %clang_cc1 -triple x86_64-unknown-unknown -emit-llvm -o - %s | FileCheck %s
// Check that PR17480 is fixed: __attribute__((used)) ignored in templated
// classes
namespace InstantiateUsedMemberDefinition {
template <typename T>
struct S {
int __attribute__((used)) f() {
return 0;
}
};
void test() {
// Check that InstantiateUsedMemberDefinition::S<int>::f() is defined
// as a result of the S class template implicit instantiation
// CHECK: define linkonce_odr i32 @_ZN31InstantiateUsedMemberDefinition1SIiE1fEv
S<int> inst;
}
} // namespace InstantiateUsedMemberDefinition