1 /* Look up a 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 dlsym_args
25 {
26   /* The arguments to dlsym_doit.  */
27   void *handle;
28   const char *name;
29   void *who;
30 
31   /* The return value of dlsym_doit.  */
32   void *sym;
33 };
34 
35 static void
dlsym_doit(void * a)36 dlsym_doit (void *a)
37 {
38   struct dlsym_args *args = (struct dlsym_args *) a;
39 
40   args->sym = _dl_sym (args->handle, args->name, args->who);
41 }
42 
43 static void *
dlsym_implementation(void * handle,const char * name,void * dl_caller)44 dlsym_implementation (void *handle, const char *name, void *dl_caller)
45 {
46   struct dlsym_args args;
47   args.who = dl_caller;
48   args.handle = handle;
49   args.name = name;
50 
51   /* Protect against concurrent loads and unloads.  */
52   __rtld_lock_lock_recursive (GL(dl_load_lock));
53 
54   void *result = (_dlerror_run (dlsym_doit, &args) ? NULL : args.sym);
55 
56   __rtld_lock_unlock_recursive (GL(dl_load_lock));
57 
58   return result;
59 }
60 
61 #ifdef SHARED
62 void *
___dlsym(void * handle,const char * name)63 ___dlsym (void *handle, const char *name)
64 {
65   if (GLRO (dl_dlfcn_hook) != NULL)
66     return GLRO (dl_dlfcn_hook)->dlsym (handle, name, RETURN_ADDRESS (0));
67   else
68     return dlsym_implementation (handle, name, RETURN_ADDRESS (0));
69 }
70 versioned_symbol (libc, ___dlsym, dlsym, GLIBC_2_34);
71 
72 # if OTHER_SHLIB_COMPAT (libdl, GLIBC_2_0, GLIBC_2_34)
73 compat_symbol (libdl, ___dlsym, dlsym, GLIBC_2_0);
74 # endif
75 
76 #else /* !SHARED */
77 /* Also used with _dlfcn_hook.  */
78 void *
__dlsym(void * handle,const char * name,void * dl_caller)79 __dlsym (void *handle, const char *name, void *dl_caller)
80 {
81   return dlsym_implementation (handle, name, dl_caller);
82 }
83 
84 void *
___dlsym(void * handle,const char * name)85 ___dlsym (void *handle, const char *name)
86 {
87   return __dlsym (handle, name, RETURN_ADDRESS (0));
88 }
89 weak_alias (___dlsym, dlsym)
90 #endif /* !SHARED */
91