diff --git a/clang/lib/CodeGen/CodeGenModule.h b/clang/lib/CodeGen/CodeGenModule.h --- a/clang/lib/CodeGen/CodeGenModule.h +++ b/clang/lib/CodeGen/CodeGenModule.h @@ -423,6 +423,10 @@ /// Global annotations. std::vector Annotations; + // Store deferred function annotations so they can be emitted at the end with + // most up to date ValueDecl that will have all the inherited annotations. + llvm::DenseMap DeferredAnnotations; + /// Map used to get unique annotation strings. llvm::StringMap AnnotationStrings; diff --git a/clang/lib/CodeGen/CodeGenModule.cpp b/clang/lib/CodeGen/CodeGenModule.cpp --- a/clang/lib/CodeGen/CodeGenModule.cpp +++ b/clang/lib/CodeGen/CodeGenModule.cpp @@ -689,6 +689,7 @@ void CodeGenModule::clear() { DeferredDeclsToEmit.clear(); EmittedDeferredDecls.clear(); + DeferredAnnotations.clear(); if (OpenMPRuntime) OpenMPRuntime->clear(); } @@ -3078,6 +3079,10 @@ } void CodeGenModule::EmitGlobalAnnotations() { + for (const auto& [F, VD] : DeferredAnnotations) + AddGlobalAnnotations(VD, F); + DeferredAnnotations.clear(); + if (Annotations.empty()) return; @@ -4316,6 +4321,11 @@ } } + // Store the declaration associated with this function so it is emitted at + // the end. + if (D && D->hasAttr() && isa(Entry)) + DeferredAnnotations[cast(Entry)] = cast(D); + if ((isa(Entry) || isa(Entry)) && (Entry->getValueType() == Ty)) { return Entry; @@ -4346,6 +4356,11 @@ llvm::Function::Create(FTy, llvm::Function::ExternalLinkage, Entry ? StringRef() : MangledName, &getModule()); + // Update the declartion associated with this function so any new or inherited + // annotations are emitted. + if (D && D->hasAttr()) + DeferredAnnotations[F] = cast(D); + // If we already created a function with the same mangled name (but different // type) before, take its name and add it to the list of functions to be // replaced with F at the end of CodeGen. @@ -4603,6 +4618,7 @@ // Lookup the entry, lazily creating it if necessary. llvm::GlobalValue *Entry = GetGlobalValue(MangledName); unsigned TargetAS = getContext().getTargetAddressSpace(AddrSpace); + if (Entry) { if (WeakRefReferences.erase(Entry)) { if (D && !D->hasAttr()) @@ -5666,8 +5682,6 @@ AddGlobalCtor(Fn, CA->getPriority()); if (const DestructorAttr *DA = D->getAttr()) AddGlobalDtor(Fn, DA->getPriority(), true); - if (D->hasAttr()) - AddGlobalAnnotations(D, Fn); } void CodeGenModule::EmitAliasDefinition(GlobalDecl GD) { diff --git a/clang/test/CodeGen/annotations-decl-use-define.c b/clang/test/CodeGen/annotations-decl-use-define.c new file mode 100644 --- /dev/null +++ b/clang/test/CodeGen/annotations-decl-use-define.c @@ -0,0 +1,16 @@ +// RUN: %clang_cc1 %s -emit-llvm -o - | FileCheck %s + +// Test annotation attributes are still emitted when the function is used before +// it is defined with annotations. + +void foo(void); +void *xxx = (void*)foo; +void __attribute__((annotate("bar"))) foo() {} + +// CHECK: target triple +// CHECK-DAG: private unnamed_addr constant [4 x i8] c"bar\00", section "llvm.metadata" + +// CHECK: @llvm.global.annotations = appending global [1 x { ptr, ptr, ptr, i32, ptr }] [{ +// CHECK-SAME: { ptr @foo, +// CHECK-SAME: }], section "llvm.metadata" + diff --git a/clang/test/CodeGen/annotations-declaration.c b/clang/test/CodeGen/annotations-declaration.c new file mode 100644 --- /dev/null +++ b/clang/test/CodeGen/annotations-declaration.c @@ -0,0 +1,17 @@ +// RUN: %clang_cc1 %s -emit-llvm -o - | FileCheck %s + +// Test annotation attributes are emitted for declarations. + +__attribute__((annotate("bar"))) int foo(); + +int main() { + return foo(); +} + +// CHECK: target triple +// CHECK-DAG: private unnamed_addr constant [4 x i8] c"bar\00", section "llvm.metadata" + +// CHECK: @llvm.global.annotations = appending global [1 x { ptr, ptr, ptr, i32, ptr }] [{ +// CHECK-SAME: { ptr @foo, +// CHECK-SAME: }], section "llvm.metadata" + diff --git a/clang/test/CodeGen/annotations-global.c b/clang/test/CodeGen/annotations-global.c --- a/clang/test/CodeGen/annotations-global.c +++ b/clang/test/CodeGen/annotations-global.c @@ -33,15 +33,15 @@ // CHECK: @llvm.global.annotations = appending global [11 x { ptr, ptr, ptr, i32, ptr }] [{ // CHECK-SAME: { ptr @a.bar, // CHECK-SAME: { ptr @a.bar, -// CHECK-SAME: { ptr @a, -// CHECK-SAME: { ptr @a, -// CHECK-SAME: { ptr @a, -// CHECK-SAME: { ptr @a, // CHECK-SAME: { ptr @sfoo, // CHECK-SAME: { ptr @sfoo, // CHECK-SAME: { ptr @foo, // CHECK-SAME: { ptr @foo, // CHECK-SAME: { ptr addrspacecast (ptr addrspace(1) @addrspace1_var to ptr), +// CHECK-SAME: { ptr @a, +// CHECK-SAME: { ptr @a, +// CHECK-SAME: { ptr @a, +// CHECK-SAME: { ptr @a, // CHECK-SAME: }], section "llvm.metadata" // AS1-GLOBALS: target datalayout = "{{.+}}-A5-G1" diff --git a/clang/test/CodeGenCXX/attr-annotate-destructor.cpp b/clang/test/CodeGenCXX/attr-annotate-destructor.cpp new file mode 100644 --- /dev/null +++ b/clang/test/CodeGenCXX/attr-annotate-destructor.cpp @@ -0,0 +1,10 @@ +// RUN: %clang_cc1 %s -S -emit-llvm -triple x86_64-unknown-linux-gnu -o - | FileCheck %s + +// Test annotation attributes on destructors doesn't crash. + +struct k { + ~k() __attribute__((annotate(""))) {} +}; +void m() { k(); } + +// CHECK: @llvm.global.annotations = appending global [2 x { ptr, ptr, ptr, i32, ptr }] [{