1 /* Post-processing of a symbol produced by dlsym, dlvsym.
2 Copyright (C) 1999-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
20 /* Return the link map containing the caller address. */
21 static struct link_map *
_dl_sym_find_caller_link_map(ElfW (Addr)caller)22 _dl_sym_find_caller_link_map (ElfW(Addr) caller)
23 {
24 struct link_map *l = _dl_find_dso_for_object (caller);
25 if (l != NULL)
26 return l;
27 else
28 /* If the address is not recognized the call comes from the main
29 program (we hope). */
30 return GL(dl_ns)[LM_ID_BASE]._ns_loaded;
31 }
32
33 /* Translates RESULT, *REF, VALUE into a symbol address from the point
34 of view of MATCH. Performs IFUNC resolution and auditing if
35 necessary. If MATCH is NULL, CALLER is used to determine it. */
36 static void *
_dl_sym_post(lookup_t result,const ElfW (Sym)* ref,void * value,ElfW (Addr)caller,struct link_map * match)37 _dl_sym_post (lookup_t result, const ElfW(Sym) *ref, void *value,
38 ElfW(Addr) caller, struct link_map *match)
39 {
40 /* Resolve indirect function address. */
41 if (__glibc_unlikely (ELFW(ST_TYPE) (ref->st_info) == STT_GNU_IFUNC))
42 {
43 DL_FIXUP_VALUE_TYPE fixup
44 = DL_FIXUP_MAKE_VALUE (result, (ElfW(Addr)) value);
45 fixup = elf_ifunc_invoke (DL_FIXUP_VALUE_ADDR (fixup));
46 value = (void *) DL_FIXUP_VALUE_CODE_ADDR (fixup);
47 }
48
49 #ifdef SHARED
50 /* Auditing checkpoint: we have a new binding. Provide the
51 auditing libraries the possibility to change the value and
52 tell us whether further auditing is wanted. */
53 if (__glibc_unlikely (GLRO(dl_naudit) > 0))
54 {
55 if (match == NULL)
56 match = _dl_sym_find_caller_link_map (caller);
57 _dl_audit_symbind_alt (match, ref, &value, result);
58 }
59 #endif
60 return value;
61 }
62