diff --git a/elf/Makefile b/elf/Makefile
index 698a6ab985..267e920561 100644
--- a/elf/Makefile
+++ b/elf/Makefile
@@ -224,7 +224,7 @@ tests += restest1 preloadtest loadfail multiload origtest resolvfail \
tst-tls-ie tst-tls-ie-dlmopen argv0test \
tst-glibc-hwcaps tst-glibc-hwcaps-prepend tst-glibc-hwcaps-mask \
tst-tls20 tst-tls21 tst-dlmopen-dlerror tst-dlmopen-gethostbyname \
- tst-dl-is_dso
+ tst-dl-is_dso tst-no-default-paths-helper
# reldep9
tests-internal += loadtest unload unload2 circleload1 \
neededtest neededtest2 neededtest3 neededtest4 \
@@ -232,7 +232,8 @@ tests-internal += loadtest unload unload2 circleload1 \
tst-ptrguard1 tst-stackguard1 tst-libc_dlvsym \
tst-create_format1 tst-tls-surplus tst-dl-hwcaps_split
tests-container += tst-pldd tst-dlopen-tlsmodid-container \
- tst-dlopen-self-container tst-preload-pthread-libc
+ tst-dlopen-self-container tst-preload-pthread-libc \
+ tst-no-default-paths-dlopen tst-no-default-paths-dlinfo
test-srcs = tst-pathopt
selinux-enabled := $(shell cat /selinux/enforce 2> /dev/null)
ifneq ($(selinux-enabled),1)
@@ -442,7 +443,8 @@ ifeq (yes,$(build-shared))
ifeq ($(run-built-tests),yes)
tests-special += $(objpfx)tst-pathopt.out $(objpfx)tst-rtld-load-self.out \
$(objpfx)tst-rtld-preload.out $(objpfx)argv0test.out \
- $(objpfx)tst-rtld-help.out
+ $(objpfx)tst-rtld-help.out \
+ $(objpfx)tst-no-default-paths-helper.out
endif
tests-special += $(objpfx)check-textrel.out $(objpfx)check-execstack.out \
$(objpfx)check-wx-segment.out \
@@ -1295,6 +1297,13 @@ tst-tst-dlopen-self-no-pie = yes
CFLAGS-tst-dlopen-self-pie.c += $(pie-ccflag)
LDFLAGS-tst-dlopen-self-container += -Wl,-rpath,\$$ORIGIN
+$(objpfx)tst-no-default-paths-dlopen.out: $(objpfx)tst-no-default-paths-helper
+$(objpfx)tst-no-default-paths-dlinfo.out: $(objpfx)tst-no-default-paths-helper
+$(objpfx)tst-no-default-paths-helper.out:
+ touch $@
+ $(evaluate-test)
+$(objpfx)tst-no-default-paths-helper: $(libdl) $(objpfx)tst-no-default-paths-helper.o
+
CFLAGS-ifuncmain1pic.c += $(pic-ccflag)
CFLAGS-ifuncmain1picstatic.c += $(pic-ccflag)
CFLAGS-ifuncmain1staticpic.c += $(pic-ccflag)
diff --git a/elf/tst-no-default-paths-dlinfo.c b/elf/tst-no-default-paths-dlinfo.c
new file mode 100644
index 0000000000..b9f534f14d
--- /dev/null
+++ b/elf/tst-no-default-paths-dlinfo.c
@@ -0,0 +1,96 @@
+/* Test that dlinfo respects --no-default-paths
+ Copyright (C) 2021 Free Software Foundation, Inc.
+ This file is part of the GNU C Library.
+
+ The GNU C Library is free software; you can redistribute it and/or
+ modify it under the terms of the GNU Lesser General Public
+ License as published by the Free Software Foundation; either
+ version 2.1 of the License, or (at your option) any later version.
+
+ The GNU C Library is distributed in the hope that it will be useful,
+ but WITHOUT ANY WARRANTY; without even the implied warranty of
+ MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
+ Lesser General Public License for more details.
+
+ You should have received a copy of the GNU Lesser General Public
+ License along with the GNU C Library; if not, see
+ . */
+
+#include
+#include
+#include
+#include
+#include
+#include
+
+const char libs_dir[] = "/tmp/tst-no-default-paths-dlinfo";
+const char elf_dir[] = "/elf";
+const char marker_lib[] = "/libmarkermod1.so";
+const char libc[] = "/" LIBC_SO;
+const char libdl[] = "/" LIBDL_SO;
+const char helper_program[] = "/tst-no-default-paths-helper";
+
+// Link system lib into libs_dir so the child process can access it
+static void
+link_system_lib_to_dir (const char *system_lib)
+{
+ char lib_src[4096] = {};
+ strcpy (lib_src, support_slibdir_prefix);
+ strcat (lib_src, system_lib);
+
+ char lib_dest[4096] = {};
+ strcpy (lib_dest, libs_dir);
+ strcat (lib_dest, system_lib);
+
+ unlink (lib_dest);
+ xsymlink (lib_src, lib_dest);
+}
+
+// Link libmarker into libs_dir also
+static void
+link_libmarker (void)
+{
+ char marker_src[4096] = {};
+ strcpy (marker_src, support_objdir_root);
+ strcat (marker_src, elf_dir);
+ strcat (marker_src, marker_lib);
+
+ char marker_dest[4096] = {};
+ strcpy (marker_dest, libs_dir);
+ strcat (marker_dest, marker_lib);
+
+ unlink (marker_dest);
+ xsymlink (marker_src, marker_dest);
+}
+
+static int
+do_test (void)
+{
+ xmkdirp (libs_dir, 0755);
+
+ link_system_lib_to_dir (libc);
+ link_system_lib_to_dir (libdl);
+ link_libmarker ();
+
+ char helper_path[4096] = {};
+ strcpy (helper_path, support_objdir_root);
+ strcat (helper_path, elf_dir);
+ strcat (helper_path, helper_program);
+
+ char *const argv[] =
+ {
+ strdup (support_container_elf_ldso_path),
+ strdup ("--no-default-paths"),
+ strdup ("--library-path"),
+ strdup (libs_dir),
+ helper_path,
+ strdup ("dlinfo"),
+ NULL
+ };
+
+ int ret = support_subprogram_wait (argv[0], argv);
+
+ return WEXITSTATUS (ret);
+}
+
+#include
diff --git a/elf/tst-no-default-paths-dlopen.c b/elf/tst-no-default-paths-dlopen.c
new file mode 100644
index 0000000000..c74beaa970
--- /dev/null
+++ b/elf/tst-no-default-paths-dlopen.c
@@ -0,0 +1,97 @@
+/* Test that --no-default-paths doesn't search system dirs
+ Copyright (C) 2021 Free Software Foundation, Inc.
+ This file is part of the GNU C Library.
+
+ The GNU C Library is free software; you can redistribute it and/or
+ modify it under the terms of the GNU Lesser General Public
+ License as published by the Free Software Foundation; either
+ version 2.1 of the License, or (at your option) any later version.
+
+ The GNU C Library is distributed in the hope that it will be useful,
+ but WITHOUT ANY WARRANTY; without even the implied warranty of
+ MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
+ Lesser General Public License for more details.
+
+ You should have received a copy of the GNU Lesser General Public
+ License along with the GNU C Library; if not, see
+ . */
+
+#include
+#include
+#include
+#include
+#include
+#include
+
+const char libs_dir[] = "/tmp/tst-no-default-paths-dlopen";
+const char elf_dir[] = "/elf";
+const char marker_lib[] = "/libmarkermod1.so";
+const char libc[] = "/" LIBC_SO;
+const char libdl[] = "/" LIBDL_SO;
+const char helper_program[] = "/tst-no-default-paths-helper";
+
+// Link system lib into libs_dir so the child process can access it
+static void
+link_system_lib_to_dir (const char *system_lib)
+{
+ char lib_src[4096] = {};
+ strcpy (lib_src, support_slibdir_prefix);
+ strcat (lib_src, system_lib);
+
+ char lib_dest[4096] = {};
+ strcpy (lib_dest, libs_dir);
+ strcat (lib_dest, system_lib);
+
+ unlink (lib_dest);
+ xsymlink (lib_src, lib_dest);
+}
+
+// Link libmarker into the system libs directory, where the child should *not*
+// be able to find it.
+static void
+link_libmarker (void)
+{
+ char marker_src[4096] = {};
+ strcpy (marker_src, support_objdir_root);
+ strcat (marker_src, elf_dir);
+ strcat (marker_src, marker_lib);
+
+ char marker_dest[4096] = {};
+ strcat (marker_dest, support_slibdir_prefix);
+ strcat (marker_dest, marker_lib);
+
+ unlink (marker_dest);
+ xsymlink (marker_src, marker_dest);
+}
+
+static int
+do_test (void)
+{
+ xmkdirp (libs_dir, 0755);
+
+ link_system_lib_to_dir (libc);
+ link_system_lib_to_dir (libdl);
+ link_libmarker ();
+
+ char helper_path[4096] = {};
+ strcpy (helper_path, support_objdir_root);
+ strcat (helper_path, elf_dir);
+ strcat (helper_path, helper_program);
+
+ char *const argv[] =
+ {
+ strdup (support_container_elf_ldso_path),
+ strdup ("--no-default-paths"),
+ strdup ("--library-path"),
+ strdup (libs_dir),
+ helper_path,
+ strdup ("dlopen"),
+ NULL
+ };
+
+ int ret = support_subprogram_wait (argv[0], argv);
+
+ return WEXITSTATUS (ret);
+}
+
+#include
diff --git a/elf/tst-no-default-paths-helper.c b/elf/tst-no-default-paths-helper.c
new file mode 100644
index 0000000000..6ccab27deb
--- /dev/null
+++ b/elf/tst-no-default-paths-helper.c
@@ -0,0 +1,69 @@
+/* Helper for --no-default-paths tests
+ Copyright (C) 2021 Free Software Foundation, Inc.
+ This file is part of the GNU C Library.
+
+ The GNU C Library is free software; you can redistribute it and/or
+ modify it under the terms of the GNU Lesser General Public
+ License as published by the Free Software Foundation; either
+ version 2.1 of the License, or (at your option) any later version.
+
+ The GNU C Library is distributed in the hope that it will be useful,
+ but WITHOUT ANY WARRANTY; without even the implied warranty of
+ MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
+ Lesser General Public License for more details.
+
+ You should have received a copy of the GNU Lesser General Public
+ License along with the GNU C Library; if not, see
+ . */
+
+#include
+#include
+#include
+#include
+#include
+#include
+
+const char marker_lib[] = "libmarkermod1.so";
+const char usage[] = "Call this as \"helper dlopen\" or \"helper dlinfo\"\n";
+
+int
+main (int argc, char *argv[])
+{
+ if (argc < 2)
+ FAIL_EXIT1 (usage);
+
+ int mode;
+ if (strcmp (argv[1], "dlopen") == 0)
+ mode = 0;
+ else if (strcmp (argv[1], "dlinfo") == 0)
+ mode = 1;
+ else
+ FAIL_EXIT1 (usage);
+
+ void *handle = dlopen ("libmarkermod1.so", RTLD_LAZY);
+ if (handle == NULL)
+ {
+ if (mode)
+ FAIL_EXIT1 ("Failed to load libmarkermod1.so\n");
+ return 0;
+ }
+ else if (handle && mode == 0)
+ FAIL_EXIT1 ("Loaded libmarkermod1.so successfully!\n");
+
+ Dl_serinfo *buffer = xmalloc (sizeof(*buffer));
+ if (dlinfo (handle, RTLD_DI_SERINFOSIZE, buffer))
+ FAIL_EXIT1 ("dlinfo failed: %s\n", dlerror());
+
+ buffer = xrealloc (buffer, buffer->dls_size);
+ if (dlinfo (handle, RTLD_DI_SERINFO, buffer))
+ FAIL_EXIT1 ("dlinfo failed: %s\n", dlerror ());
+
+ // Checking if two paths are equal is tricky, so just check that there's
+ // exactly one and that it starts with /tmp, which should be sufficient to
+ // prove that the system paths have been excluded.
+ TEST_VERIFY_EXIT (buffer->dls_cnt == 1);
+ const char *name = buffer->dls_serpath[0].dls_name;
+ TEST_VERIFY_EXIT (strstr (name, "/tmp") == name);
+
+ return 0;
+}
diff --git a/support/Makefile b/support/Makefile
index 5c69f0de4b..9730f6b7dc 100644
--- a/support/Makefile
+++ b/support/Makefile
@@ -212,6 +212,7 @@ CFLAGS-support_paths.c = \
-DSRCDIR_PATH=\"`cd .. ; pwd`\" \
-DOBJDIR_PATH=\"`cd $(objpfx)/..; pwd`\" \
-DOBJDIR_ELF_LDSO_PATH=\"`cd $(objpfx)/..; pwd`/elf/$(rtld-installed-name)\" \
+ -DCONTAINER_ELF_LDSO_PATH=\"$(rtlddir)/$(rtld-installed-name)\" \
-DINSTDIR_PATH=\"$(prefix)\" \
-DLIBDIR_PATH=\"$(libdir)\" \
-DBINDIR_PATH=\"$(bindir)\" \
diff --git a/support/support.h b/support/support.h
index 9ec8ecb8d7..21f994067d 100644
--- a/support/support.h
+++ b/support/support.h
@@ -108,6 +108,10 @@ extern const char support_objdir_root[];
e.g. OBJDIR_PATH/elf/ld-linux-x86-64.so.2 */
extern const char support_objdir_elf_ldso[];
+/* Corresponds to the path to the runtime linker for containerised tests,
+ e.g. /lib64/ld-linux-x86-64.so.2 */
+extern const char support_container_elf_ldso_path[];
+
/* Corresponds to the --prefix= passed to configure. */
extern const char support_install_prefix[];
/* Corresponds to the install's lib/ or lib64/ directory. */
diff --git a/support/support_paths.c b/support/support_paths.c
index d18e71e38b..4a14d4e78e 100644
--- a/support/support_paths.c
+++ b/support/support_paths.c
@@ -44,6 +44,12 @@ const char support_objdir_elf_ldso[] = OBJDIR_ELF_LDSO_PATH;
# error please -DOBJDIR_ELF_LDSO_PATH=something in the Makefile
#endif
+#ifdef CONTAINER_ELF_LDSO_PATH
+const char support_container_elf_ldso_path[] = CONTAINER_ELF_LDSO_PATH;
+#else
+# error please -DCONTAINER_ELF_LDSO_PATH=something in the Makefile
+#endif
+
#ifdef INSTDIR_PATH
/* Corresponds to the --prefix= passed to configure. */
const char support_install_prefix[] = INSTDIR_PATH;