llvm-project/clang/test/CodeGenCXX/extern-c.cpp

Ignoring revisions in .git-blame-ignore-revs. Click here to bypass and see the normal blame view.

91 lines
2.1 KiB
C++
Raw Normal View History

// RUN: %clang_cc1 -triple %itanium_abi_triple -emit-llvm %s -o - | FileCheck %s
namespace foo {
// CHECK-NOT: @a = global
extern "C" int a;
// CHECK-NOT: @_ZN3foo1bE = global
extern int b;
// CHECK: @_ZN3foo1cE = {{(dso_local )?}}global
int c = 5;
// CHECK-NOT: @_ZN3foo1dE
extern "C" struct d;
// CHECK-NOT: should_not_appear
extern "C++" int should_not_appear;
// CHECK: @_ZN3foo10extern_cxxE = {{(dso_local )?}}global
extern "C++" int extern_cxx = 0;
}
// CHECK-NOT: @global_a = {{(dso_local )?}}global
extern "C" int global_a;
// CHECK: @global_b = {{(dso_local )?}}global
extern "C" int global_b = 0;
// CHECK-NOT: should_not_appear
extern "C++" int should_not_appear;
// CHECK: @extern_cxx = {{(dso_local )?}}global
extern "C++" int extern_cxx = 0;
namespace test1 {
namespace {
struct X {};
}
extern "C" {
// CHECK: @test1_b = {{(dso_local )?}}global
X test1_b = X();
}
void *use = &test1_b;
// CHECK: @_ZN5test13useE = {{(dso_local )?}}global
}
namespace test2 {
namespace {
struct X {};
}
// CHECK: @test2_b = {{(dso_local )?}}global
extern "C" X test2_b;
X test2_b;
}
extern "C" {
static int unused_var;
static int unused_fn() { return 0; }
__attribute__((used)) static int internal_var;
__attribute__((used)) static int internal_fn() { return 0; }
__attribute__((used)) static int duplicate_internal_var;
__attribute__((used)) static int duplicate_internal_fn() { return 0; }
namespace N {
__attribute__((used)) static int duplicate_internal_var;
__attribute__((used)) static int duplicate_internal_fn() { return 0; }
}
Change some addUsedGlobal to addUsedOrCompilerUsedGlobal An global value in the `llvm.used` list does not have GC root semantics on ELF targets. This will be changed in a subsequent backend patch. Change some `llvm.used` in the ELF code path to use `llvm.compiler.used` to prevent undesired GC root semantics. Change one extern "C" alias (due to `__attribute__((used))` in extern "C") to use `llvm.compiler.used` on all targets. GNU ld has a rule "`__start_/__stop_` references from a live input section retain the associated C identifier name sections", which LLD may drop entirely (currently refined to exclude SHF_LINK_ORDER/SHF_GROUP) in a future release (the rule makes it clumsy to GC metadata sections; D96914 added a way to try the potential future behavior). For `llvm.used` global values defined in a C identifier name section, keep using `llvm.used` so that the future LLD change will not affect them. rnk kindly categorized the changes: ``` ObjC/blocks: this wants GC root semantics, since ObjC mainly runs on Mac. MS C++ ABI stuff: wants GC root semantics, no change OpenMP: unsure, but GC root semantics probably don't hurt CodeGenModule: affected in this patch to *not* use GC root semantics so that __attribute__((used)) behavior remains the same on ELF, plus two other minor use cases that don't want GC semantics Coverage: Probably want GC root semantics CGExpr.cpp: refers to LTO, wants GC root CGDeclCXX.cpp: one is MS ABI specific, so yes GC root, one is some other C++ init functionality, which should form GC roots (C++ initializers can have side effects and must run) CGDecl.cpp: Changed in this patch for __attribute__((used)) ``` Differential Revision: https://reviews.llvm.org/D97446
2021-02-27 02:42:07 +08:00
// CHECK: @llvm.compiler.used = appending global {{.*}} @internal_var {{.*}} @internal_fn
// CHECK-NOT: @unused
// CHECK-NOT: @duplicate_internal
// CHECK: @internal_var = internal alias i32, i32* @_ZL12internal_var
// CHECK-NOT: @unused
// CHECK-NOT: @duplicate_internal
// CHECK: @internal_fn = internal alias i32 (), i32 ()* @_ZL11internal_fnv
// CHECK-NOT: @unused
// CHECK-NOT: @duplicate_internal
}
namespace PR19411 {
struct A { void f(); };
extern "C" void A::f() { void g(); g(); }
// CHECK-LABEL: @_ZN7PR194111A1fEv(
// CHECK: call {{.*}}void @g()
}