From: Richard Trieu Date: Tue, 23 Oct 2018 01:26:28 +0000 (+0000) Subject: [CodeGen] Attach InlineHint to more functions X-Git-Url: https://granicus.if.org/sourcecode?a=commitdiff_plain;h=8377fd8a5d4e0722e2e7c155cfd7649dd4fc2ce9;p=clang [CodeGen] Attach InlineHint to more functions For instantiated functions, search the template pattern to see if it marked inline to determine if InlineHint attribute should be added to the function. git-svn-id: https://llvm.org/svn/llvm-project/cfe/trunk@344987 91177308-0d34-0410-b5e6-96231b3b80d8 --- diff --git a/lib/CodeGen/CodeGenModule.cpp b/lib/CodeGen/CodeGenModule.cpp index c8e672a567..8faaed1ec9 100644 --- a/lib/CodeGen/CodeGenModule.cpp +++ b/lib/CodeGen/CodeGenModule.cpp @@ -1299,9 +1299,19 @@ void CodeGenModule::SetLLVMFunctionAttributesForDefinition(const Decl *D, // Otherwise, propagate the inline hint attribute and potentially use its // absence to mark things as noinline. if (auto *FD = dyn_cast(D)) { - if (any_of(FD->redecls(), [&](const FunctionDecl *Redecl) { - return Redecl->isInlineSpecified(); - })) { + // Search function and template pattern redeclarations for inline. + auto CheckForInline = [](const FunctionDecl *FD) { + auto CheckRedeclForInline = [](const FunctionDecl *Redecl) { + return Redecl->isInlineSpecified(); + }; + if (any_of(FD->redecls(), CheckRedeclForInline)) + return true; + const FunctionDecl *Pattern = FD->getTemplateInstantiationPattern(); + if (!Pattern) + return false; + return any_of(Pattern->redecls(), CheckRedeclForInline); + }; + if (CheckForInline(FD)) { B.addAttribute(llvm::Attribute::InlineHint); } else if (CodeGenOpts.getInlining() == CodeGenOptions::OnlyHintInlining && diff --git a/test/CodeGenCXX/inline-template-hint.cpp b/test/CodeGenCXX/inline-template-hint.cpp new file mode 100644 index 0000000000..3e64b1f749 --- /dev/null +++ b/test/CodeGenCXX/inline-template-hint.cpp @@ -0,0 +1,34 @@ +// RUN: %clang_cc1 %s -std=c++11 -triple=x86_64-linux -O2 \ +// RUN: -finline-functions -emit-llvm -disable-llvm-passes -o - \ +// RUN: | FileCheck -allow-deprecated-dag-overlap %s \ +// RUN: --check-prefix=CHECK --check-prefix=SUITABLE +// RUN: %clang_cc1 %s -std=c++11 -triple=x86_64-linux -O2 \ +// RUN: -finline-hint-functions -emit-llvm -disable-llvm-passes -o - \ +// RUN: | FileCheck -allow-deprecated-dag-overlap %s \ +// RUN: --check-prefix=CHECK --check-prefix=HINTED +// RUN: %clang_cc1 %s -std=c++11 -triple=x86_64-linux -O2 \ +// RUN: -fno-inline -emit-llvm -disable-llvm-passes -o - \ +// RUN: | FileCheck -allow-deprecated-dag-overlap %s \ +// RUN: --check-prefix=CHECK --check-prefix=NOINLINE + +struct A { + inline void int_run(int); + + template + inline void template_run(T); +}; + +// CHECK: @_ZN1A7int_runEi({{.*}}) [[ATTR:#[0-9]+]] +void A::int_run(int) {} +// CHECK: @_ZN1A12template_runIiEEvT_({{.*}}) [[ATTR]] +template +void A::template_run(T) {} + +void bar() { + A().int_run(1); + A().template_run(1); +} + +// SUITABLE: attributes [[ATTR]] = { {{.*}}inlinehint{{.*}} } +// HINTED: attributes [[ATTR]] = { {{.*}}inlinehint{{.*}} } +// NOINLINE: attributes [[ATTR]] = { {{.*}}noinline{{.*}} }