Blame libdw/dwarf_getpubnames.c

Packit Service 97d2fb
/* Get public symbol information.
Packit Service 97d2fb
   Copyright (C) 2002, 2003, 2004, 2005, 2008 Red Hat, Inc.
Packit Service 97d2fb
   This file is part of elfutils.
Packit Service 97d2fb
   Written by Ulrich Drepper <drepper@redhat.com>, 2002.
Packit Service 97d2fb
Packit Service 97d2fb
   This file is free software; you can redistribute it and/or modify
Packit Service 97d2fb
   it under the terms of either
Packit Service 97d2fb
Packit Service 97d2fb
     * the GNU Lesser General Public License as published by the Free
Packit Service 97d2fb
       Software Foundation; either version 3 of the License, or (at
Packit Service 97d2fb
       your option) any later version
Packit Service 97d2fb
Packit Service 97d2fb
   or
Packit Service 97d2fb
Packit Service 97d2fb
     * the GNU General Public License as published by the Free
Packit Service 97d2fb
       Software Foundation; either version 2 of the License, or (at
Packit Service 97d2fb
       your option) any later version
Packit Service 97d2fb
Packit Service 97d2fb
   or both in parallel, as here.
Packit Service 97d2fb
Packit Service 97d2fb
   elfutils is distributed in the hope that it will be useful, but
Packit Service 97d2fb
   WITHOUT ANY WARRANTY; without even the implied warranty of
Packit Service 97d2fb
   MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
Packit Service 97d2fb
   General Public License for more details.
Packit Service 97d2fb
Packit Service 97d2fb
   You should have received copies of the GNU General Public License and
Packit Service 97d2fb
   the GNU Lesser General Public License along with this program.  If
Packit Service 97d2fb
   not, see <http://www.gnu.org/licenses/>.  */
Packit Service 97d2fb
Packit Service 97d2fb
#ifdef HAVE_CONFIG_H
Packit Service 97d2fb
# include <config.h>
Packit Service 97d2fb
#endif
Packit Service 97d2fb
Packit Service 97d2fb
#include <assert.h>
Packit Service 97d2fb
#include <stdlib.h>
Packit Service 97d2fb
#include <string.h>
Packit Service 97d2fb
Packit Service 97d2fb
#include <libdwP.h>
Packit Service 97d2fb
#include <dwarf.h>
Packit Service 97d2fb
#include <system.h>
Packit Service 97d2fb
Packit Service 97d2fb
Packit Service 97d2fb
static int
Packit Service 97d2fb
get_offsets (Dwarf *dbg)
Packit Service 97d2fb
{
Packit Service 97d2fb
  size_t allocated = 0;
Packit Service 97d2fb
  size_t cnt = 0;
Packit Service 97d2fb
  struct pubnames_s *mem = NULL;
Packit Service 97d2fb
  const size_t entsize = sizeof (struct pubnames_s);
Packit Service 97d2fb
  unsigned char *const startp = dbg->sectiondata[IDX_debug_pubnames]->d_buf;
Packit Service 97d2fb
  unsigned char *readp = startp;
Packit Service 97d2fb
  unsigned char *endp = readp + dbg->sectiondata[IDX_debug_pubnames]->d_size;
Packit Service 97d2fb
Packit Service 97d2fb
  while (readp + 14 < endp)
Packit Service 97d2fb
    {
Packit Service 97d2fb
      /* If necessary, allocate more entries.  */
Packit Service 97d2fb
      if (cnt >= allocated)
Packit Service 97d2fb
	{
Packit Service 97d2fb
	  allocated = MAX (10, 2 * allocated);
Packit Service 97d2fb
	  struct pubnames_s *newmem
Packit Service 97d2fb
	    = (struct pubnames_s *) realloc (mem, allocated * entsize);
Packit Service 97d2fb
	  if (newmem == NULL)
Packit Service 97d2fb
	    {
Packit Service 97d2fb
	      __libdw_seterrno (DWARF_E_NOMEM);
Packit Service 97d2fb
	    err_return:
Packit Service 97d2fb
	      free (mem);
Packit Service 97d2fb
	      return -1;
Packit Service 97d2fb
	    }
Packit Service 97d2fb
Packit Service 97d2fb
	  mem = newmem;
Packit Service 97d2fb
	}
Packit Service 97d2fb
Packit Service 97d2fb
      /* Read the set header.  */
Packit Service 97d2fb
      int len_bytes = 4;
Packit Service 97d2fb
      Dwarf_Off len = read_4ubyte_unaligned_inc (dbg, readp);
Packit Service 97d2fb
      if (len == DWARF3_LENGTH_64_BIT)
Packit Service 97d2fb
	{
Packit Service 97d2fb
	  len = read_8ubyte_unaligned_inc (dbg, readp);
Packit Service 97d2fb
	  len_bytes = 8;
Packit Service 97d2fb
	}
Packit Service 97d2fb
      else if (unlikely (len >= DWARF3_LENGTH_MIN_ESCAPE_CODE
Packit Service 97d2fb
			 && len <= DWARF3_LENGTH_MAX_ESCAPE_CODE))
Packit Service 97d2fb
	{
Packit Service 97d2fb
	  __libdw_seterrno (DWARF_E_INVALID_DWARF);
Packit Service 97d2fb
	  goto err_return;
Packit Service 97d2fb
	}
Packit Service 97d2fb
Packit Service 97d2fb
      /* Now we know the offset of the first offset/name pair.  */
Packit Service 97d2fb
      mem[cnt].set_start = readp + 2 + 2 * len_bytes - startp;
Packit Service 97d2fb
      mem[cnt].address_len = len_bytes;
Packit Service 97d2fb
      size_t max_size = dbg->sectiondata[IDX_debug_pubnames]->d_size;
Packit Service 97d2fb
      if (mem[cnt].set_start >= max_size
Packit Service 97d2fb
	  || len - (2 + 2 * len_bytes) > max_size - mem[cnt].set_start)
Packit Service 97d2fb
	/* Something wrong, the first entry is beyond the end of
Packit Service 97d2fb
	   the section.  Or the length of the whole unit is too big.  */
Packit Service 97d2fb
	break;
Packit Service 97d2fb
Packit Service 97d2fb
      /* Read the version.  It better be two for now.  */
Packit Service 97d2fb
      uint16_t version = read_2ubyte_unaligned (dbg, readp);
Packit Service 97d2fb
      if (unlikely (version != 2))
Packit Service 97d2fb
	{
Packit Service 97d2fb
	  __libdw_seterrno (DWARF_E_INVALID_VERSION);
Packit Service 97d2fb
	  goto err_return;
Packit Service 97d2fb
	}
Packit Service 97d2fb
Packit Service 97d2fb
      /* Get the CU offset.  */
Packit Service 97d2fb
      if (__libdw_read_offset (dbg, dbg, IDX_debug_pubnames,
Packit Service 97d2fb
			       readp + 2, len_bytes,
Packit Service 97d2fb
			       &mem[cnt].cu_offset, IDX_debug_info, 3))
Packit Service 97d2fb
	/* Error has been already set in reader.  */
Packit Service 97d2fb
	goto err_return;
Packit Service 97d2fb
Packit Service 97d2fb
      /* Determine the size of the CU header.  */
Packit Service 97d2fb
      unsigned char *infop
Packit Service 97d2fb
	= ((unsigned char *) dbg->sectiondata[IDX_debug_info]->d_buf
Packit Service 97d2fb
	   + mem[cnt].cu_offset);
Packit Service 97d2fb
      if (read_4ubyte_unaligned_noncvt (infop) == DWARF3_LENGTH_64_BIT)
Packit Service 97d2fb
	mem[cnt].cu_header_size = 23;
Packit Service 97d2fb
      else
Packit Service 97d2fb
	mem[cnt].cu_header_size = 11;
Packit Service 97d2fb
Packit Service 97d2fb
      ++cnt;
Packit Service 97d2fb
Packit Service 97d2fb
      /* Advance to the next set.  */
Packit Service 97d2fb
      readp += len;
Packit Service 97d2fb
    }
Packit Service 97d2fb
Packit Service 97d2fb
  if (mem == NULL || cnt == 0)
Packit Service 97d2fb
    {
Packit Service 97d2fb
      free (mem);
Packit Service 97d2fb
      __libdw_seterrno (DWARF_E_NO_ENTRY);
Packit Service 97d2fb
      return -1;
Packit Service 97d2fb
    }
Packit Service 97d2fb
Packit Service 97d2fb
  dbg->pubnames_sets = (struct pubnames_s *) realloc (mem, cnt * entsize);
Packit Service 97d2fb
  dbg->pubnames_nsets = cnt;
Packit Service 97d2fb
Packit Service 97d2fb
  return 0;
Packit Service 97d2fb
}
Packit Service 97d2fb
Packit Service 97d2fb
Packit Service 97d2fb
ptrdiff_t
Packit Service 97d2fb
dwarf_getpubnames (Dwarf *dbg,
Packit Service 97d2fb
		   int (*callback) (Dwarf *, Dwarf_Global *, void *),
Packit Service 97d2fb
		   void *arg, ptrdiff_t offset)
Packit Service 97d2fb
{
Packit Service 97d2fb
  if (dbg == NULL)
Packit Service 97d2fb
    return -1l;
Packit Service 97d2fb
Packit Service 97d2fb
  if (unlikely (offset < 0))
Packit Service 97d2fb
    {
Packit Service 97d2fb
      __libdw_seterrno (DWARF_E_INVALID_OFFSET);
Packit Service 97d2fb
      return -1l;
Packit Service 97d2fb
    }
Packit Service 97d2fb
Packit Service 97d2fb
  /* Make sure it is a valid offset.  */
Packit Service 97d2fb
  if (unlikely (dbg->sectiondata[IDX_debug_pubnames] == NULL
Packit Service 97d2fb
		|| ((size_t) offset
Packit Service 97d2fb
		    >= dbg->sectiondata[IDX_debug_pubnames]->d_size)))
Packit Service 97d2fb
    /* No (more) entry.  */
Packit Service 97d2fb
    return 0;
Packit Service 97d2fb
Packit Service 97d2fb
  /* If necessary read the set information.  */
Packit Service 97d2fb
  if (dbg->pubnames_nsets == 0 && unlikely (get_offsets (dbg) != 0))
Packit Service 97d2fb
    return -1l;
Packit Service 97d2fb
Packit Service 97d2fb
  /* Find the place where to start.  */
Packit Service 97d2fb
  size_t cnt;
Packit Service 97d2fb
  if (offset == 0)
Packit Service 97d2fb
    {
Packit Service 97d2fb
      cnt = 0;
Packit Service 97d2fb
      offset = dbg->pubnames_sets[0].set_start;
Packit Service 97d2fb
    }
Packit Service 97d2fb
  else
Packit Service 97d2fb
    {
Packit Service 97d2fb
      for (cnt = 0; cnt + 1 < dbg->pubnames_nsets; ++cnt)
Packit Service 97d2fb
	if ((Dwarf_Off) offset >= dbg->pubnames_sets[cnt].set_start)
Packit Service 97d2fb
	  {
Packit Service 97d2fb
	    assert ((Dwarf_Off) offset
Packit Service 97d2fb
		    < dbg->pubnames_sets[cnt + 1].set_start);
Packit Service 97d2fb
	    break;
Packit Service 97d2fb
	  }
Packit Service 97d2fb
      assert (cnt + 1 < dbg->pubnames_nsets);
Packit Service 97d2fb
    }
Packit Service 97d2fb
Packit Service 97d2fb
  unsigned char *startp
Packit Service 97d2fb
    = (unsigned char *) dbg->sectiondata[IDX_debug_pubnames]->d_buf;
Packit Service 97d2fb
  unsigned char *endp
Packit Service 97d2fb
    = startp + dbg->sectiondata[IDX_debug_pubnames]->d_size;
Packit Service 97d2fb
  unsigned char *readp = startp + offset;
Packit Service 97d2fb
  while (1)
Packit Service 97d2fb
    {
Packit Service 97d2fb
      Dwarf_Global gl;
Packit Service 97d2fb
Packit Service 97d2fb
      gl.cu_offset = (dbg->pubnames_sets[cnt].cu_offset
Packit Service 97d2fb
		      + dbg->pubnames_sets[cnt].cu_header_size);
Packit Service 97d2fb
Packit Service 97d2fb
      while (1)
Packit Service 97d2fb
	{
Packit Service 97d2fb
	  /* READP points to the next offset/name pair.  */
Packit Service 97d2fb
	  if (readp + dbg->pubnames_sets[cnt].address_len > endp)
Packit Service 97d2fb
	    goto invalid_dwarf;
Packit Service 97d2fb
	  if (dbg->pubnames_sets[cnt].address_len == 4)
Packit Service 97d2fb
	    gl.die_offset = read_4ubyte_unaligned_inc (dbg, readp);
Packit Service 97d2fb
	  else
Packit Service 97d2fb
	    gl.die_offset = read_8ubyte_unaligned_inc (dbg, readp);
Packit Service 97d2fb
Packit Service 97d2fb
	  /* If the offset is zero we reached the end of the set.  */
Packit Service 97d2fb
	  if (gl.die_offset == 0)
Packit Service 97d2fb
	    break;
Packit Service 97d2fb
Packit Service 97d2fb
	  /* Add the CU offset.  */
Packit Service 97d2fb
	  gl.die_offset += dbg->pubnames_sets[cnt].cu_offset;
Packit Service 97d2fb
Packit Service 97d2fb
	  gl.name = (char *) readp;
Packit Service 97d2fb
	  readp = (unsigned char *) memchr (gl.name, '\0', endp - readp);
Packit Service 97d2fb
	  if (unlikely (readp == NULL))
Packit Service 97d2fb
	    {
Packit Service 97d2fb
	    invalid_dwarf:
Packit Service 97d2fb
	      __libdw_seterrno (DWARF_E_INVALID_DWARF);
Packit Service 97d2fb
	      return -1l;
Packit Service 97d2fb
	    }
Packit Service 97d2fb
	  readp++;
Packit Service 97d2fb
Packit Service 97d2fb
	  /* We found name and DIE offset.  Report it.  */
Packit Service 97d2fb
	  if (callback (dbg, &gl, arg) != DWARF_CB_OK)
Packit Service 97d2fb
	    {
Packit Service 97d2fb
	      /* The user wants us to stop.  Return the offset of the
Packit Service 97d2fb
		 next entry.  */
Packit Service 97d2fb
	      return readp - startp;
Packit Service 97d2fb
	    }
Packit Service 97d2fb
	}
Packit Service 97d2fb
Packit Service 97d2fb
      if (++cnt == dbg->pubnames_nsets)
Packit Service 97d2fb
	/* This was the last set.  */
Packit Service 97d2fb
	break;
Packit Service 97d2fb
Packit Service 97d2fb
      startp = (unsigned char *) dbg->sectiondata[IDX_debug_pubnames]->d_buf;
Packit Service 97d2fb
      readp = startp + dbg->pubnames_sets[cnt].set_start;
Packit Service 97d2fb
    }
Packit Service 97d2fb
Packit Service 97d2fb
  /* We are done.  No more entries.  */
Packit Service 97d2fb
  return 0;
Packit Service 97d2fb
}