`clang -fpic -fno-semantic-interposition` may set dso_local on variables for -fpic. GCC folks consider there are 'address interposition' and 'semantic interposition', and 'disabling semantic interposition' can optimize function calls but cannot change variable references to use local aliases (https://gcc.gnu.org/bugzilla/show_bug.cgi?id=100483). This patch removes dso_local for variables in `clang -fpic -fno-semantic-interposition` mode so that the built shared objects can work with copy relocations. Building llvm-project tiself with -fno-semantic-interposition (D102453) should now be safe with trunk Clang. Example: ``` // a.c int var; int *addr() { return var; } // old: cannot be interposed movslq .Lvar$local(%rip), %rax // new: can be interposed movq var@GOTPCREL(%rip), %rax movslq (%rax), %rax ``` The local alias lowering for `GlobalVariable`s is kept in case there is a future option allowing local aliases. Reviewed By: rnk Differential Revision: https://reviews.llvm.org/D102583
39 lines
1.6 KiB
C
39 lines
1.6 KiB
C
/// -fno-semantic-interposition is the default and local aliases (via dso_local) are allowed.
|
|
// RUN: %clang_cc1 -triple x86_64-linux-gnu -emit-llvm -mrelocation-model pic -pic-level 1 %s -o - | FileCheck %s --check-prefixes=CHECK,NOMETADATA
|
|
|
|
/// -fsemantic-interposition sets a module metadata.
|
|
// RUN: %clang_cc1 -triple x86_64-linux-gnu -emit-llvm -mrelocation-model pic -pic-level 1 -fsemantic-interposition %s -o - | FileCheck %s --check-prefixes=PREEMPT,METADATA
|
|
|
|
/// Traditional half-baked behavior: interprocedural optimizations are allowed
|
|
/// but local aliases are not used.
|
|
// RUN: %clang_cc1 -triple x86_64-linux-gnu -emit-llvm -mrelocation-model pic -pic-level 1 -fhalf-no-semantic-interposition %s -o - | FileCheck %s --check-prefixes=PREEMPT,NOMETADATA
|
|
|
|
// CHECK: @var = global i32 0, align 4
|
|
// CHECK: @ext_var = external global i32, align 4
|
|
// CHECK: @ifunc = ifunc i32 (), bitcast (i8* ()* @ifunc_resolver to i32 ()*)
|
|
// CHECK: define dso_local i32 @func()
|
|
// CHECK: declare i32 @ext()
|
|
|
|
// PREEMPT: @var = global i32 0, align 4
|
|
// PREEMPT: @ext_var = external global i32, align 4
|
|
// PREEMPT: @ifunc = ifunc i32 (), bitcast (i8* ()* @ifunc_resolver to i32 ()*)
|
|
// PREEMPT: define i32 @func()
|
|
// PREEMPT: declare i32 @ext()
|
|
|
|
// METADATA: !{{[0-9]+}} = !{i32 1, !"SemanticInterposition", i32 1}
|
|
// NOMETADATA-NOT: "SemanticInterposition"
|
|
|
|
int var;
|
|
extern int ext_var;
|
|
|
|
int ifunc(void) __attribute__((ifunc("ifunc_resolver")));
|
|
|
|
int func(void) { return 0; }
|
|
int ext(void);
|
|
|
|
static void *ifunc_resolver() { return func; }
|
|
|
|
int foo() {
|
|
return var + ext_var + ifunc() + func() + ext();
|
|
}
|