LCOV - code coverage report
Current view: top level - libdwfl - dwfl_module_getsym.c (source / functions) Hit Total Coverage
Test: elfutils-0.191 Lines: 89 95 93.7 %
Date: 2024-07-23 19:52:03 Functions: 3 3 100.0 %
Legend: Lines: hit not hit | Branches: + taken - not taken # not executed Branches: 66 76 86.8 %

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

Generated by: LCOV version 1.16