Skip to content

Commit e2e82c9

Browse files
committed
[CodeGenModule] Drop dso_local on function declarations for ELF -fno-pic -fno-direct-access-external-data
ELF -fno-pic sets dso_local on a function declaration to allow direct accesses when taking its address (similar to a data symbol). The emitted code follows the traditional GCC/Clang -fno-pic behavior: an absolute relocation is produced. If the function is not defined in the executable, a canonical PLT entry will be needed at link time. This is similar to a copy relocation and is incompatible with (-Bsymbolic or --dynamic-list linked shared objects / protected symbols in a shared object). This patch gives -fno-pic code a way to avoid such a canonical PLT entry. The FIXME was about a generalization for -fpie -mpie-copy-relocations (now -fpie -fdirect-access-external-data). While we could set dso_local to avoid GOT when taking the address of a function declaration (there is an ignorable difference about R_386_PC32 vs R_386_PLT32 on i386), it likely does not provide any benefit and can just cause trouble, so we don't make the generalization.
1 parent b02eab9 commit e2e82c9

File tree

2 files changed

+21
-10
lines changed

2 files changed

+21
-10
lines changed

clang/lib/CodeGen/CodeGenModule.cpp

+19-8
Original file line numberDiff line numberDiff line change
@@ -983,16 +983,27 @@ static bool shouldAssumeDSOLocal(const CodeGenModule &CGM,
983983
if (TT.isPPC64())
984984
return false;
985985

986-
// If we can use copy relocations we can assume it is local.
987-
if (auto *Var = dyn_cast<llvm::GlobalVariable>(GV))
988-
if (!Var->isThreadLocal() && CGOpts.DirectAccessExternalData)
986+
if (CGOpts.DirectAccessExternalData) {
987+
// If -fdirect-access-external-data (default for -fno-pic), set dso_local
988+
// for non-thread-local variables. If the symbol is not defined in the
989+
// executable, a copy relocation will be needed at link time. dso_local is
990+
// excluded for thread-local variables because they generally don't support
991+
// copy relocations.
992+
if (auto *Var = dyn_cast<llvm::GlobalVariable>(GV))
993+
if (!Var->isThreadLocal())
994+
return true;
995+
996+
// -fno-pic sets dso_local on a function declaration to allow direct
997+
// accesses when taking its address (similar to a data symbol). If the
998+
// function is not defined in the executable, a canonical PLT entry will be
999+
// needed at link time. -fno-direct-access-external-data can avoid the
1000+
// canonical PLT entry. We don't generalize this condition to -fpie/-fpic as
1001+
// it could just cause trouble without providing perceptible benefits.
1002+
if (isa<llvm::Function>(GV) && !CGOpts.NoPLT && RM == llvm::Reloc::Static)
9891003
return true;
1004+
}
9901005

991-
// If we can use a plt entry as the symbol address we can assume it
992-
// is local.
993-
// FIXME: This should work for PIE, but the gold linker doesn't support it.
994-
if (isa<llvm::Function>(GV) && !CGOpts.NoPLT && RM == llvm::Reloc::Static)
995-
return true;
1006+
// If we can use copy relocations we can assume it is local.
9961007

9971008
// Otherwise don't assume it is local.
9981009
return false;

clang/test/CodeGen/dso-local-executable.c

+2-2
Original file line numberDiff line numberDiff line change
@@ -43,9 +43,9 @@
4343
// STATIC-INDIRECT-NEXT: @bar = external global i32
4444
// STATIC-INDIRECT-NEXT: @local_thread_var = dso_local thread_local global i32 42
4545
// STATIC-INDIRECT-NEXT: @thread_var = external thread_local global i32
46-
// STATIC-INDIRECT-DAG: declare dso_local void @import_func()
46+
// STATIC-INDIRECT-DAG: declare void @import_func()
4747
// STATIC-INDIRECT-DAG: define dso_local i32* @zed()
48-
// STATIC-INDIRECT-DAG: declare dso_local void @foo()
48+
// STATIC-INDIRECT-DAG: declare void @foo()
4949

5050
// RUN: %clang_cc1 -triple x86_64 -emit-llvm -pic-level 1 -pic-is-pie %s -o - | FileCheck --check-prefix=PIE %s
5151
// PIE: @baz = dso_local global i32 42

0 commit comments

Comments
 (0)