Index: ELF/Driver.h =================================================================== --- ELF/Driver.h +++ ELF/Driver.h @@ -27,7 +27,7 @@ class LinkerDriver { public: void main(ArrayRef Args, bool CanExitEarly); - void addFile(StringRef Path); + void addFile(StringRef Path, bool SharedFoundFromSearchPaths = false); void addLibrary(StringRef Name); private: Index: ELF/Driver.cpp =================================================================== --- ELF/Driver.cpp +++ ELF/Driver.cpp @@ -153,7 +153,7 @@ // Opens and parses a file. Path has to be resolved already. // Newly created memory buffers are owned by this driver. -void LinkerDriver::addFile(StringRef Path) { +void LinkerDriver::addFile(StringRef Path, bool SharedFoundFromSearchPaths) { using namespace sys::fs; Optional Buffer = readFile(Path); @@ -184,6 +184,7 @@ return; } Files.push_back(createSharedFile(MBRef)); + Files.back()->SharedFoundFromSearchPaths = SharedFoundFromSearchPaths; return; default: if (InLib) @@ -196,7 +197,7 @@ // Add a given library by searching it from input search paths. void LinkerDriver::addLibrary(StringRef Name) { if (Optional Path = searchLibrary(Name)) - addFile(*Path); + addFile(*Path, /*SharedFoundFromSearchPaths=*/true); else error("unable to find library -l" + Name); } Index: ELF/InputFiles.h =================================================================== --- ELF/InputFiles.h +++ ELF/InputFiles.h @@ -93,6 +93,10 @@ uint16_t EMachine = llvm::ELF::EM_NONE; uint8_t OSABI = 0; + // True if this is a shared library found in library search paths, as opposed + // to having explicit path. + bool SharedFoundFromSearchPaths; + // Cache for toString(). Only toString() should use this member. mutable std::string ToStringCache; Index: ELF/InputFiles.cpp =================================================================== --- ELF/InputFiles.cpp +++ ELF/InputFiles.cpp @@ -661,7 +661,12 @@ // DSOs are identified by soname, and they usually contain // DT_SONAME tag in their header. But if they are missing, // filenames are used as default sonames. - SoName = sys::path::filename(this->getName()); + // If the library is found at an explicitly given path and does not have a + // soname, hardcode that path in soname. Such libraries should not require + // RPATH or LD_LIBRARY_PATH to run. + SoName = this->SharedFoundFromSearchPaths + ? sys::path::filename(this->getName()) + : this->getName(); if (!DynamicSec) return; Index: test/ELF/as-needed-no-reloc.s =================================================================== --- test/ELF/as-needed-no-reloc.s +++ test/ELF/as-needed-no-reloc.s @@ -16,7 +16,7 @@ # CHECK-NEXT: Other: 0 # CHECK-NEXT: Section: Undefined -# CHECK: NEEDED SharedLibrary (as-needed-no-reloc{{.*}}2.so) +# CHECK: NEEDED SharedLibrary ({{.*}}as-needed-no-reloc{{.*}}2.so) .globl _start _start: Index: test/ELF/no-soname.s =================================================================== --- /dev/null +++ test/ELF/no-soname.s @@ -0,0 +1,31 @@ +// RUN: llvm-mc -filetype=obj -triple=x86_64-pc-linux %s -o %t.o +// RUN: mkdir -p %T/no-soname +// RUN: ld.lld %t.o -shared -o %T/no-soname/libfoo.so + +// RUN: ld.lld %t.o %T/no-soname/libfoo.so -o %t +// RUN: llvm-readobj --dynamic-table %t | FileCheck %s + +// CHECK: 0x0000000000000001 NEEDED SharedLibrary ({{.*}}/no-soname/libfoo.so) +// CHECK-NOT: NEEDED + +// RUN: ld.lld %t.o %T/no-soname/../no-soname/libfoo.so -o %t +// RUN: llvm-readobj --dynamic-table %t | FileCheck %s --check-prefix=CHECK2 + +// CHECK2: 0x0000000000000001 NEEDED SharedLibrary ({{.*}}/no-soname/../no-soname/libfoo.so) +// CHECK2-NOT: NEEDED + +// RUN: ld.lld %t.o -L%T/no-soname/../no-soname -lfoo -o %t +// RUN: llvm-readobj --dynamic-table %t | FileCheck %s --check-prefix=CHECK3 + +// CHECK3: 0x0000000000000001 NEEDED SharedLibrary (libfoo.so) +// CHECK3-NOT: NEEDED + +// RUN: ld.lld %t.o -shared -soname libbar.so -o %T/no-soname/libbar.so +// RUN: ld.lld %t.o %T/no-soname/libbar.so -o %t +// RUN: llvm-readobj --dynamic-table %t | FileCheck %s --check-prefix=CHECK4 + +// CHECK4: 0x0000000000000001 NEEDED SharedLibrary (libbar.so) +// CHECK4-NOT: NEEDED + +.global _start +_start: Index: test/ELF/relro-omagic.s =================================================================== --- test/ELF/relro-omagic.s +++ test/ELF/relro-omagic.s @@ -10,16 +10,20 @@ # NORELRO-NEXT: 0 00000000 0000000000000000 # NORELRO-NEXT: 1 .dynsym 00000048 0000000000200120 # NORELRO-NEXT: 2 .hash 00000020 0000000000200168 -# NORELRO-NEXT: 3 .dynstr 00000021 0000000000200188 -# NORELRO-NEXT: 4 .rela.dyn 00000018 00000000002001b0 -# NORELRO-NEXT: 5 .rela.plt 00000018 00000000002001c8 -# NORELRO-NEXT: 6 .text 0000000a 00000000002001e0 TEXT DATA -# NORELRO-NEXT: 7 .plt 00000020 00000000002001f0 TEXT DATA -# NORELRO-NEXT: 8 .data 00000008 0000000000200210 DATA -# NORELRO-NEXT: 9 .foo 00000004 0000000000200218 DATA -# NORELRO-NEXT: 10 .dynamic 000000f0 0000000000200220 -# NORELRO-NEXT: 11 .got 00000008 0000000000200310 DATA -# NORELRO-NEXT: 12 .got.plt 00000020 0000000000200318 DATA +# NORELRO-NEXT: 3 .dynstr 0000004c 0000000000200188 +# NORELRO-NEXT: 4 .rela.dyn 00000018 00000000002001d8 +# NORELRO-NEXT: 5 .rela.plt 00000018 00000000002001f0 +# NORELRO-NEXT: 6 .text 0000000a 0000000000200208 TEXT DATA +# NORELRO-NEXT: 7 .plt 00000020 0000000000200220 TEXT DATA +# NORELRO-NEXT: 8 .data 00000008 0000000000200240 DATA +# NORELRO-NEXT: 9 .foo 00000004 0000000000200248 DATA +# NORELRO-NEXT: 10 .dynamic 000000f0 0000000000200250 +# NORELRO-NEXT: 11 .got 00000008 0000000000200340 DATA +# NORELRO-NEXT: 12 .got.plt 00000020 0000000000200348 DATA +# NORELRO-NEXT: 13 .comment 00000008 0000000000000000 +# NORELRO-NEXT: 14 .symtab 00000060 0000000000000000 +# NORELRO-NEXT: 15 .shstrtab 0000007b 0000000000000000 +# NORELRO-NEXT: 16 .strtab 00000013 0000000000000000 # NOPHDRS: ProgramHeaders [ # NOPHDRS-NOT: PT_GNU_RELRO