x86: In ld.so, diagnose missing APX support in APX-only builds
[glibc.git] / dlfcn / dlopen.c
blob3775da3295faa725a5b0692ada4d9d8e7baa11e3
1 /* Load a shared object at run time.
2 Copyright (C) 1995-2024 Free Software Foundation, Inc.
3 This file is part of the GNU C Library.
5 The GNU C Library is free software; you can redistribute it and/or
6 modify it under the terms of the GNU Lesser General Public
7 License as published by the Free Software Foundation; either
8 version 2.1 of the License, or (at your option) any later version.
10 The GNU C Library is distributed in the hope that it will be useful,
11 but WITHOUT ANY WARRANTY; without even the implied warranty of
12 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
13 Lesser General Public License for more details.
15 You should have received a copy of the GNU Lesser General Public
16 License along with the GNU C Library; if not, see
17 <https://www.gnu.org/licenses/>. */
19 #include <dlfcn.h>
20 #include <libintl.h>
21 #include <stddef.h>
22 #include <unistd.h>
23 #include <ldsodefs.h>
24 #include <shlib-compat.h>
26 struct dlopen_args
28 /* The arguments for dlopen_doit. */
29 const char *file;
30 int mode;
31 /* The return value of dlopen_doit. */
32 void *new;
33 /* Address of the caller. */
34 const void *caller;
38 /* Non-shared code has no support for multiple namespaces. */
39 #ifdef SHARED
40 # define NS __LM_ID_CALLER
41 #else
42 # define NS LM_ID_BASE
43 #endif
46 static void
47 dlopen_doit (void *a)
49 struct dlopen_args *args = (struct dlopen_args *) a;
51 if (args->mode & ~(RTLD_BINDING_MASK | RTLD_NOLOAD | RTLD_DEEPBIND
52 | RTLD_GLOBAL | RTLD_LOCAL | RTLD_NODELETE
53 | __RTLD_SPROF))
54 _dl_signal_error (0, NULL, NULL, _("invalid mode parameter"));
56 args->new = GLRO(dl_open) (args->file ?: "", args->mode | __RTLD_DLOPEN,
57 args->caller,
58 args->file == NULL ? LM_ID_BASE : NS,
59 __libc_argc, __libc_argv, __environ);
63 static void *
64 dlopen_implementation (const char *file, int mode, void *dl_caller)
66 struct dlopen_args args;
67 args.file = file;
68 args.mode = mode;
69 args.caller = dl_caller;
71 return _dlerror_run (dlopen_doit, &args) ? NULL : args.new;
74 #ifdef SHARED
75 void *
76 ___dlopen (const char *file, int mode)
78 if (GLRO (dl_dlfcn_hook) != NULL)
79 return GLRO (dl_dlfcn_hook)->dlopen (file, mode, RETURN_ADDRESS (0));
80 else
81 return dlopen_implementation (file, mode, RETURN_ADDRESS (0));
83 versioned_symbol (libc, ___dlopen, dlopen, GLIBC_2_34);
85 # if OTHER_SHLIB_COMPAT (libdl, GLIBC_2_1, GLIBC_2_34)
86 compat_symbol (libdl, ___dlopen, dlopen, GLIBC_2_1);
87 # endif
88 #else /* !SHARED */
89 /* Also used with _dlfcn_hook. */
90 void *
91 __dlopen (const char *file, int mode, void *dl_caller)
93 return dlopen_implementation (file, mode, dl_caller);
96 void *
97 ___dlopen (const char *file, int mode)
99 return __dlopen (file, mode, RETURN_ADDRESS (0));
101 weak_alias (___dlopen, dlopen)
102 static_link_warning (dlopen)
103 #endif /* !SHARED */