xref: /aosp_15_r20/external/elfutils/libdwfl/dwfl_module_getsym.c (revision 7304104da70ce23c86437a01be71edd1a2d7f37e)
1*7304104dSAndroid Build Coastguard Worker /* Find debugging and symbol information for a module in libdwfl.
2*7304104dSAndroid Build Coastguard Worker    Copyright (C) 2006-2014 Red Hat, Inc.
3*7304104dSAndroid Build Coastguard Worker    This file is part of elfutils.
4*7304104dSAndroid Build Coastguard Worker 
5*7304104dSAndroid Build Coastguard Worker    This file is free software; you can redistribute it and/or modify
6*7304104dSAndroid Build Coastguard Worker    it under the terms of either
7*7304104dSAndroid Build Coastguard Worker 
8*7304104dSAndroid Build Coastguard Worker      * the GNU Lesser General Public License as published by the Free
9*7304104dSAndroid Build Coastguard Worker        Software Foundation; either version 3 of the License, or (at
10*7304104dSAndroid Build Coastguard Worker        your option) any later version
11*7304104dSAndroid Build Coastguard Worker 
12*7304104dSAndroid Build Coastguard Worker    or
13*7304104dSAndroid Build Coastguard Worker 
14*7304104dSAndroid Build Coastguard Worker      * the GNU General Public License as published by the Free
15*7304104dSAndroid Build Coastguard Worker        Software Foundation; either version 2 of the License, or (at
16*7304104dSAndroid Build Coastguard Worker        your option) any later version
17*7304104dSAndroid Build Coastguard Worker 
18*7304104dSAndroid Build Coastguard Worker    or both in parallel, as here.
19*7304104dSAndroid Build Coastguard Worker 
20*7304104dSAndroid Build Coastguard Worker    elfutils is distributed in the hope that it will be useful, but
21*7304104dSAndroid Build Coastguard Worker    WITHOUT ANY WARRANTY; without even the implied warranty of
22*7304104dSAndroid Build Coastguard Worker    MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
23*7304104dSAndroid Build Coastguard Worker    General Public License for more details.
24*7304104dSAndroid Build Coastguard Worker 
25*7304104dSAndroid Build Coastguard Worker    You should have received copies of the GNU General Public License and
26*7304104dSAndroid Build Coastguard Worker    the GNU Lesser General Public License along with this program.  If
27*7304104dSAndroid Build Coastguard Worker    not, see <http://www.gnu.org/licenses/>.  */
28*7304104dSAndroid Build Coastguard Worker 
29*7304104dSAndroid Build Coastguard Worker #ifdef HAVE_CONFIG_H
30*7304104dSAndroid Build Coastguard Worker # include <config.h>
31*7304104dSAndroid Build Coastguard Worker #endif
32*7304104dSAndroid Build Coastguard Worker 
33*7304104dSAndroid Build Coastguard Worker #include "libdwflP.h"
34*7304104dSAndroid Build Coastguard Worker 
35*7304104dSAndroid Build Coastguard Worker const char *
36*7304104dSAndroid Build Coastguard Worker internal_function
__libdwfl_getsym(Dwfl_Module * mod,int ndx,GElf_Sym * sym,GElf_Addr * addr,GElf_Word * shndxp,Elf ** elfp,Dwarf_Addr * biasp,bool * resolved,bool adjust_st_value)37*7304104dSAndroid Build Coastguard Worker __libdwfl_getsym (Dwfl_Module *mod, int ndx, GElf_Sym *sym, GElf_Addr *addr,
38*7304104dSAndroid Build Coastguard Worker 		  GElf_Word *shndxp, Elf **elfp, Dwarf_Addr *biasp,
39*7304104dSAndroid Build Coastguard Worker 		  bool *resolved, bool adjust_st_value)
40*7304104dSAndroid Build Coastguard Worker {
41*7304104dSAndroid Build Coastguard Worker   if (unlikely (mod == NULL))
42*7304104dSAndroid Build Coastguard Worker     return NULL;
43*7304104dSAndroid Build Coastguard Worker 
44*7304104dSAndroid Build Coastguard Worker   if (unlikely (mod->symdata == NULL))
45*7304104dSAndroid Build Coastguard Worker     {
46*7304104dSAndroid Build Coastguard Worker       int result = INTUSE(dwfl_module_getsymtab) (mod);
47*7304104dSAndroid Build Coastguard Worker       if (result < 0)
48*7304104dSAndroid Build Coastguard Worker 	return NULL;
49*7304104dSAndroid Build Coastguard Worker     }
50*7304104dSAndroid Build Coastguard Worker 
51*7304104dSAndroid Build Coastguard Worker   /* All local symbols should come before all global symbols.  If we
52*7304104dSAndroid Build Coastguard Worker      have an auxiliary table make sure all the main locals come first,
53*7304104dSAndroid Build Coastguard Worker      then all aux locals, then all main globals and finally all aux globals.
54*7304104dSAndroid Build Coastguard Worker      And skip the auxiliary table zero undefined entry.  */
55*7304104dSAndroid Build Coastguard Worker   GElf_Word shndx;
56*7304104dSAndroid Build Coastguard Worker   int tndx = ndx;
57*7304104dSAndroid Build Coastguard Worker   int skip_aux_zero = (mod->syments > 0 && mod->aux_syments > 0) ? 1 : 0;
58*7304104dSAndroid Build Coastguard Worker   Elf *elf;
59*7304104dSAndroid Build Coastguard Worker   Elf_Data *symdata;
60*7304104dSAndroid Build Coastguard Worker   Elf_Data *symxndxdata;
61*7304104dSAndroid Build Coastguard Worker   Elf_Data *symstrdata;
62*7304104dSAndroid Build Coastguard Worker   if (mod->aux_symdata == NULL
63*7304104dSAndroid Build Coastguard Worker       || ndx < mod->first_global)
64*7304104dSAndroid Build Coastguard Worker     {
65*7304104dSAndroid Build Coastguard Worker       /* main symbol table (locals).  */
66*7304104dSAndroid Build Coastguard Worker       tndx = ndx;
67*7304104dSAndroid Build Coastguard Worker       elf = mod->symfile->elf;
68*7304104dSAndroid Build Coastguard Worker       symdata = mod->symdata;
69*7304104dSAndroid Build Coastguard Worker       symxndxdata = mod->symxndxdata;
70*7304104dSAndroid Build Coastguard Worker       symstrdata = mod->symstrdata;
71*7304104dSAndroid Build Coastguard Worker     }
72*7304104dSAndroid Build Coastguard Worker   else if (ndx < mod->first_global + mod->aux_first_global - skip_aux_zero)
73*7304104dSAndroid Build Coastguard Worker     {
74*7304104dSAndroid Build Coastguard Worker       /* aux symbol table (locals).  */
75*7304104dSAndroid Build Coastguard Worker       tndx = ndx - mod->first_global + skip_aux_zero;
76*7304104dSAndroid Build Coastguard Worker       elf = mod->aux_sym.elf;
77*7304104dSAndroid Build Coastguard Worker       symdata = mod->aux_symdata;
78*7304104dSAndroid Build Coastguard Worker       symxndxdata = mod->aux_symxndxdata;
79*7304104dSAndroid Build Coastguard Worker       symstrdata = mod->aux_symstrdata;
80*7304104dSAndroid Build Coastguard Worker     }
81*7304104dSAndroid Build Coastguard Worker   else if ((size_t) ndx < mod->syments + mod->aux_first_global - skip_aux_zero)
82*7304104dSAndroid Build Coastguard Worker     {
83*7304104dSAndroid Build Coastguard Worker       /* main symbol table (globals).  */
84*7304104dSAndroid Build Coastguard Worker       tndx = ndx - mod->aux_first_global + skip_aux_zero;
85*7304104dSAndroid Build Coastguard Worker       elf = mod->symfile->elf;
86*7304104dSAndroid Build Coastguard Worker       symdata = mod->symdata;
87*7304104dSAndroid Build Coastguard Worker       symxndxdata = mod->symxndxdata;
88*7304104dSAndroid Build Coastguard Worker       symstrdata = mod->symstrdata;
89*7304104dSAndroid Build Coastguard Worker     }
90*7304104dSAndroid Build Coastguard Worker   else
91*7304104dSAndroid Build Coastguard Worker     {
92*7304104dSAndroid Build Coastguard Worker       /* aux symbol table (globals).  */
93*7304104dSAndroid Build Coastguard Worker       tndx = ndx - mod->syments + skip_aux_zero;
94*7304104dSAndroid Build Coastguard Worker       elf = mod->aux_sym.elf;
95*7304104dSAndroid Build Coastguard Worker       symdata = mod->aux_symdata;
96*7304104dSAndroid Build Coastguard Worker       symxndxdata = mod->aux_symxndxdata;
97*7304104dSAndroid Build Coastguard Worker       symstrdata = mod->aux_symstrdata;
98*7304104dSAndroid Build Coastguard Worker     }
99*7304104dSAndroid Build Coastguard Worker   sym = gelf_getsymshndx (symdata, symxndxdata, tndx, sym, &shndx);
100*7304104dSAndroid Build Coastguard Worker 
101*7304104dSAndroid Build Coastguard Worker   if (unlikely (sym == NULL))
102*7304104dSAndroid Build Coastguard Worker     {
103*7304104dSAndroid Build Coastguard Worker       __libdwfl_seterrno (DWFL_E_LIBELF);
104*7304104dSAndroid Build Coastguard Worker       return NULL;
105*7304104dSAndroid Build Coastguard Worker     }
106*7304104dSAndroid Build Coastguard Worker 
107*7304104dSAndroid Build Coastguard Worker   if (sym->st_shndx != SHN_XINDEX)
108*7304104dSAndroid Build Coastguard Worker     shndx = sym->st_shndx;
109*7304104dSAndroid Build Coastguard Worker 
110*7304104dSAndroid Build Coastguard Worker   /* Figure out whether this symbol points into an SHF_ALLOC section.  */
111*7304104dSAndroid Build Coastguard Worker   bool alloc = true;
112*7304104dSAndroid Build Coastguard Worker   if ((shndxp != NULL || mod->e_type != ET_REL)
113*7304104dSAndroid Build Coastguard Worker       && (sym->st_shndx == SHN_XINDEX
114*7304104dSAndroid Build Coastguard Worker 	  || (sym->st_shndx < SHN_LORESERVE && sym->st_shndx != SHN_UNDEF)))
115*7304104dSAndroid Build Coastguard Worker     {
116*7304104dSAndroid Build Coastguard Worker       GElf_Shdr shdr_mem;
117*7304104dSAndroid Build Coastguard Worker       GElf_Shdr *shdr = gelf_getshdr (elf_getscn (elf, shndx), &shdr_mem);
118*7304104dSAndroid Build Coastguard Worker       alloc = unlikely (shdr == NULL) || (shdr->sh_flags & SHF_ALLOC);
119*7304104dSAndroid Build Coastguard Worker     }
120*7304104dSAndroid Build Coastguard Worker 
121*7304104dSAndroid Build Coastguard Worker   /* In case of an value in an allocated section the main Elf Ebl
122*7304104dSAndroid Build Coastguard Worker      might know where the real value is (e.g. for function
123*7304104dSAndroid Build Coastguard Worker      descriptors).  */
124*7304104dSAndroid Build Coastguard Worker 
125*7304104dSAndroid Build Coastguard Worker   char *ident;
126*7304104dSAndroid Build Coastguard Worker   GElf_Addr st_value = sym->st_value & ebl_func_addr_mask (mod->ebl);
127*7304104dSAndroid Build Coastguard Worker   *resolved = false;
128*7304104dSAndroid Build Coastguard Worker   if (! adjust_st_value && mod->e_type != ET_REL && alloc
129*7304104dSAndroid Build Coastguard Worker       && (GELF_ST_TYPE (sym->st_info) == STT_FUNC
130*7304104dSAndroid Build Coastguard Worker 	  || (GELF_ST_TYPE (sym->st_info) == STT_GNU_IFUNC
131*7304104dSAndroid Build Coastguard Worker 	      && (ident = elf_getident (elf, NULL)) != NULL
132*7304104dSAndroid Build Coastguard Worker 	      && ident[EI_OSABI] == ELFOSABI_LINUX)))
133*7304104dSAndroid Build Coastguard Worker     {
134*7304104dSAndroid Build Coastguard Worker       if (likely (__libdwfl_module_getebl (mod) == DWFL_E_NOERROR))
135*7304104dSAndroid Build Coastguard Worker 	{
136*7304104dSAndroid Build Coastguard Worker 	  if (elf != mod->main.elf)
137*7304104dSAndroid Build Coastguard Worker 	    {
138*7304104dSAndroid Build Coastguard Worker 	      st_value = dwfl_adjusted_st_value (mod, elf, st_value);
139*7304104dSAndroid Build Coastguard Worker 	      st_value = dwfl_deadjust_st_value (mod, mod->main.elf, st_value);
140*7304104dSAndroid Build Coastguard Worker 	    }
141*7304104dSAndroid Build Coastguard Worker 
142*7304104dSAndroid Build Coastguard Worker 	  *resolved = ebl_resolve_sym_value (mod->ebl, &st_value);
143*7304104dSAndroid Build Coastguard Worker 	  if (! *resolved)
144*7304104dSAndroid Build Coastguard Worker 	    st_value = sym->st_value;
145*7304104dSAndroid Build Coastguard Worker 	}
146*7304104dSAndroid Build Coastguard Worker     }
147*7304104dSAndroid Build Coastguard Worker 
148*7304104dSAndroid Build Coastguard Worker   if (shndxp != NULL)
149*7304104dSAndroid Build Coastguard Worker     /* Yield -1 in case of a non-SHF_ALLOC section.  */
150*7304104dSAndroid Build Coastguard Worker     *shndxp = alloc ? shndx : (GElf_Word) -1;
151*7304104dSAndroid Build Coastguard Worker 
152*7304104dSAndroid Build Coastguard Worker   switch (sym->st_shndx)
153*7304104dSAndroid Build Coastguard Worker     {
154*7304104dSAndroid Build Coastguard Worker     case SHN_ABS:		/* XXX sometimes should use bias?? */
155*7304104dSAndroid Build Coastguard Worker     case SHN_UNDEF:
156*7304104dSAndroid Build Coastguard Worker     case SHN_COMMON:
157*7304104dSAndroid Build Coastguard Worker       break;
158*7304104dSAndroid Build Coastguard Worker 
159*7304104dSAndroid Build Coastguard Worker     default:
160*7304104dSAndroid Build Coastguard Worker       if (mod->e_type == ET_REL)
161*7304104dSAndroid Build Coastguard Worker 	{
162*7304104dSAndroid Build Coastguard Worker 	  /* In an ET_REL file, the symbol table values are relative
163*7304104dSAndroid Build Coastguard Worker 	     to the section, not to the module's load base.  */
164*7304104dSAndroid Build Coastguard Worker 	  size_t symshstrndx = SHN_UNDEF;
165*7304104dSAndroid Build Coastguard Worker 	  Dwfl_Error result = __libdwfl_relocate_value (mod, elf,
166*7304104dSAndroid Build Coastguard Worker 							&symshstrndx,
167*7304104dSAndroid Build Coastguard Worker 							shndx, &st_value);
168*7304104dSAndroid Build Coastguard Worker 	  if (unlikely (result != DWFL_E_NOERROR))
169*7304104dSAndroid Build Coastguard Worker 	    {
170*7304104dSAndroid Build Coastguard Worker 	      __libdwfl_seterrno (result);
171*7304104dSAndroid Build Coastguard Worker 	      return NULL;
172*7304104dSAndroid Build Coastguard Worker 	    }
173*7304104dSAndroid Build Coastguard Worker 	}
174*7304104dSAndroid Build Coastguard Worker       else if (alloc)
175*7304104dSAndroid Build Coastguard Worker 	/* Apply the bias to the symbol value.  */
176*7304104dSAndroid Build Coastguard Worker 	st_value = dwfl_adjusted_st_value (mod,
177*7304104dSAndroid Build Coastguard Worker 					   *resolved ? mod->main.elf : elf,
178*7304104dSAndroid Build Coastguard Worker 					   st_value);
179*7304104dSAndroid Build Coastguard Worker       break;
180*7304104dSAndroid Build Coastguard Worker     }
181*7304104dSAndroid Build Coastguard Worker 
182*7304104dSAndroid Build Coastguard Worker   if (adjust_st_value)
183*7304104dSAndroid Build Coastguard Worker     sym->st_value = st_value;
184*7304104dSAndroid Build Coastguard Worker 
185*7304104dSAndroid Build Coastguard Worker   if (addr != NULL)
186*7304104dSAndroid Build Coastguard Worker     *addr = st_value;
187*7304104dSAndroid Build Coastguard Worker 
188*7304104dSAndroid Build Coastguard Worker   if (unlikely (sym->st_name >= symstrdata->d_size))
189*7304104dSAndroid Build Coastguard Worker     {
190*7304104dSAndroid Build Coastguard Worker       __libdwfl_seterrno (DWFL_E_BADSTROFF);
191*7304104dSAndroid Build Coastguard Worker       return NULL;
192*7304104dSAndroid Build Coastguard Worker     }
193*7304104dSAndroid Build Coastguard Worker   if (elfp)
194*7304104dSAndroid Build Coastguard Worker     *elfp = elf;
195*7304104dSAndroid Build Coastguard Worker   if (biasp)
196*7304104dSAndroid Build Coastguard Worker     *biasp = dwfl_adjusted_st_value (mod, elf, 0);
197*7304104dSAndroid Build Coastguard Worker   return (const char *) symstrdata->d_buf + sym->st_name;
198*7304104dSAndroid Build Coastguard Worker }
199*7304104dSAndroid Build Coastguard Worker 
200*7304104dSAndroid Build Coastguard Worker const char *
dwfl_module_getsym_info(Dwfl_Module * mod,int ndx,GElf_Sym * sym,GElf_Addr * addr,GElf_Word * shndxp,Elf ** elfp,Dwarf_Addr * bias)201*7304104dSAndroid Build Coastguard Worker dwfl_module_getsym_info (Dwfl_Module *mod, int ndx,
202*7304104dSAndroid Build Coastguard Worker 			 GElf_Sym *sym, GElf_Addr *addr,
203*7304104dSAndroid Build Coastguard Worker 			 GElf_Word *shndxp,
204*7304104dSAndroid Build Coastguard Worker 			 Elf **elfp, Dwarf_Addr *bias)
205*7304104dSAndroid Build Coastguard Worker {
206*7304104dSAndroid Build Coastguard Worker   bool resolved;
207*7304104dSAndroid Build Coastguard Worker   return __libdwfl_getsym (mod, ndx, sym, addr, shndxp, elfp, bias,
208*7304104dSAndroid Build Coastguard Worker 			   &resolved, false);
209*7304104dSAndroid Build Coastguard Worker }
INTDEF(dwfl_module_getsym_info)210*7304104dSAndroid Build Coastguard Worker INTDEF (dwfl_module_getsym_info)
211*7304104dSAndroid Build Coastguard Worker 
212*7304104dSAndroid Build Coastguard Worker const char *
213*7304104dSAndroid Build Coastguard Worker dwfl_module_getsym (Dwfl_Module *mod, int ndx,
214*7304104dSAndroid Build Coastguard Worker 		    GElf_Sym *sym, GElf_Word *shndxp)
215*7304104dSAndroid Build Coastguard Worker {
216*7304104dSAndroid Build Coastguard Worker   bool resolved;
217*7304104dSAndroid Build Coastguard Worker   return __libdwfl_getsym (mod, ndx, sym, NULL, shndxp, NULL, NULL,
218*7304104dSAndroid Build Coastguard Worker 			   &resolved, true);
219*7304104dSAndroid Build Coastguard Worker }
220*7304104dSAndroid Build Coastguard Worker INTDEF (dwfl_module_getsym)
221