1 /* Look up a versioned symbol in a shared object loaded by `dlopen'.
2 Copyright (C) 1995-2022 Free Software Foundation, Inc.
3 This file is part of the GNU C Library.
4
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.
9
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.
14
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/>. */
18
19 #include <dlfcn.h>
20 #include <ldsodefs.h>
21 #include <shlib-compat.h>
22 #include <stddef.h>
23
24 struct dlvsym_args
25 {
26 /* The arguments to dlvsym_doit. */
27 void *handle;
28 const char *name;
29 const char *version;
30 void *who;
31
32 /* The return values of dlvsym_doit. */
33 void *sym;
34 };
35
36 static void
dlvsym_doit(void * a)37 dlvsym_doit (void *a)
38 {
39 struct dlvsym_args *args = (struct dlvsym_args *) a;
40
41 args->sym = _dl_vsym (args->handle, args->name, args->version, args->who);
42 }
43
44 static void *
dlvsym_implementation(void * handle,const char * name,const char * version,void * dl_caller)45 dlvsym_implementation (void *handle, const char *name, const char *version,
46 void *dl_caller)
47 {
48 struct dlvsym_args args;
49 args.who = dl_caller;
50 args.handle = handle;
51 args.name = name;
52 args.version = version;
53
54 /* Protect against concurrent loads and unloads. */
55 __rtld_lock_lock_recursive (GL(dl_load_lock));
56
57 void *result = (_dlerror_run (dlvsym_doit, &args) ? NULL : args.sym);
58
59 __rtld_lock_unlock_recursive (GL(dl_load_lock));
60
61 return result;
62 }
63
64 #ifdef SHARED
65 void *
___dlvsym(void * handle,const char * name,const char * version)66 ___dlvsym (void *handle, const char *name, const char *version)
67 {
68 if (GLRO (dl_dlfcn_hook) != NULL)
69 return GLRO (dl_dlfcn_hook)->dlvsym (handle, name, version,
70 RETURN_ADDRESS (0));
71 else
72 return dlvsym_implementation (handle, name, version, RETURN_ADDRESS (0));
73 }
74 versioned_symbol (libc, ___dlvsym, dlvsym, GLIBC_2_34);
75
76 # if OTHER_SHLIB_COMPAT (libdl, GLIBC_2_1, GLIBC_2_34)
77 compat_symbol (libdl, ___dlvsym, dlvsym, GLIBC_2_1);
78 # endif
79
80 #else /* !SHARED */
81 /* Also used with _dlfcn_hook. */
82 void *
__dlvsym(void * handle,const char * name,const char * version,void * dl_caller)83 __dlvsym (void *handle, const char *name, const char *version, void *dl_caller)
84 {
85 return dlvsym_implementation (handle, name, version, dl_caller);
86 }
87
88 void *
___dlvsym(void * handle,const char * name,const char * version)89 ___dlvsym (void *handle, const char *name, const char *version)
90 {
91 return __dlvsym (handle, name, version, RETURN_ADDRESS (0));
92 }
93 weak_alias (___dlvsym, dlvsym)
94 #endif /* !SHARED */
95