#ifdef HAVE_CONFIG_H
# include <config.h>
#endif
#include "libdwflP.h"
const char *
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)
{
if (unlikely (mod == NULL))
return NULL;
if (unlikely (mod->symdata == NULL))
{
int result = INTUSE(dwfl_module_getsymtab) (mod);
if (result < 0)
return NULL;
}
GElf_Word shndx;
int tndx = ndx;
int skip_aux_zero = (mod->syments > 0 && mod->aux_syments > 0) ? 1 : 0;
Elf *elf;
Elf_Data *symdata;
Elf_Data *symxndxdata;
Elf_Data *symstrdata;
if (mod->aux_symdata == NULL
|| ndx < mod->first_global)
{
tndx = ndx;
elf = mod->symfile->elf;
symdata = mod->symdata;
symxndxdata = mod->symxndxdata;
symstrdata = mod->symstrdata;
}
else if (ndx < mod->first_global + mod->aux_first_global - skip_aux_zero)
{
tndx = ndx - mod->first_global + skip_aux_zero;
elf = mod->aux_sym.elf;
symdata = mod->aux_symdata;
symxndxdata = mod->aux_symxndxdata;
symstrdata = mod->aux_symstrdata;
}
else if ((size_t) ndx < mod->syments + mod->aux_first_global - skip_aux_zero)
{
tndx = ndx - mod->aux_first_global + skip_aux_zero;
elf = mod->symfile->elf;
symdata = mod->symdata;
symxndxdata = mod->symxndxdata;
symstrdata = mod->symstrdata;
}
else
{
tndx = ndx - mod->syments + skip_aux_zero;
elf = mod->aux_sym.elf;
symdata = mod->aux_symdata;
symxndxdata = mod->aux_symxndxdata;
symstrdata = mod->aux_symstrdata;
}
sym = gelf_getsymshndx (symdata, symxndxdata, tndx, sym, &shndx);
if (unlikely (sym == NULL))
{
__libdwfl_seterrno (DWFL_E_LIBELF);
return NULL;
}
if (sym->st_shndx != SHN_XINDEX)
shndx = sym->st_shndx;
bool alloc = true;
if ((shndxp != NULL || mod->e_type != ET_REL)
&& (sym->st_shndx == SHN_XINDEX
|| (sym->st_shndx < SHN_LORESERVE && sym->st_shndx != SHN_UNDEF)))
{
GElf_Shdr shdr_mem;
GElf_Shdr *shdr = gelf_getshdr (elf_getscn (elf, shndx), &shdr_mem);
alloc = unlikely (shdr == NULL) || (shdr->sh_flags & SHF_ALLOC);
}
char *ident;
GElf_Addr st_value = sym->st_value & ebl_func_addr_mask (mod->ebl);
*resolved = false;
if (! adjust_st_value && mod->e_type != ET_REL && alloc
&& (GELF_ST_TYPE (sym->st_info) == STT_FUNC
|| (GELF_ST_TYPE (sym->st_info) == STT_GNU_IFUNC
&& (ident = elf_getident (elf, NULL)) != NULL
&& ident[EI_OSABI] == ELFOSABI_LINUX)))
{
if (likely (__libdwfl_module_getebl (mod) == DWFL_E_NOERROR))
{
if (elf != mod->main.elf)
{
st_value = dwfl_adjusted_st_value (mod, elf, st_value);
st_value = dwfl_deadjust_st_value (mod, mod->main.elf, st_value);
}
*resolved = ebl_resolve_sym_value (mod->ebl, &st_value);
if (! *resolved)
st_value = sym->st_value;
}
}
if (shndxp != NULL)
*shndxp = alloc ? shndx : (GElf_Word) -1;
switch (sym->st_shndx)
{
case SHN_ABS:
case SHN_UNDEF:
case SHN_COMMON:
break;
default:
if (mod->e_type == ET_REL)
{
size_t symshstrndx = SHN_UNDEF;
Dwfl_Error result = __libdwfl_relocate_value (mod, elf,
&symshstrndx,
shndx, &st_value);
if (unlikely (result != DWFL_E_NOERROR))
{
__libdwfl_seterrno (result);
return NULL;
}
}
else if (alloc)
st_value = dwfl_adjusted_st_value (mod,
*resolved ? mod->main.elf : elf,
st_value);
break;
}
if (adjust_st_value)
sym->st_value = st_value;
if (addr != NULL)
*addr = st_value;
if (unlikely (sym->st_name >= symstrdata->d_size))
{
__libdwfl_seterrno (DWFL_E_BADSTROFF);
return NULL;
}
if (elfp)
*elfp = elf;
if (biasp)
*biasp = dwfl_adjusted_st_value (mod, elf, 0);
return (const char *) symstrdata->d_buf + sym->st_name;
}
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)
{
bool resolved;
return __libdwfl_getsym (mod, ndx, sym, addr, shndxp, elfp, bias,
&resolved, false);
}
INTDEF (dwfl_module_getsym_info)
const char *
dwfl_module_getsym (Dwfl_Module *mod, int ndx,
GElf_Sym *sym, GElf_Word *shndxp)
{
bool resolved;
return __libdwfl_getsym (mod, ndx, sym, NULL, shndxp, NULL, NULL,
&resolved, true);
}
INTDEF (dwfl_module_getsym)