Blame libdw/dwarf_formref_die.c

Packit Service 97d2fb
/* Look up the DIE in a reference-form attribute.
Packit Service 97d2fb
   Copyright (C) 2005-2010, 2018 Red Hat, Inc.
Packit Service 97d2fb
   This file is part of elfutils.
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 <string.h>
Packit Service 97d2fb
#include "libdwP.h"
Packit Service 97d2fb
#include <dwarf.h>
Packit Service 97d2fb
Packit Service 97d2fb
Packit Service 97d2fb
Dwarf_Die *
Packit Service 97d2fb
dwarf_formref_die (Dwarf_Attribute *attr, Dwarf_Die *result)
Packit Service 97d2fb
{
Packit Service 97d2fb
  if (attr == NULL)
Packit Service 97d2fb
    return NULL;
Packit Service 97d2fb
Packit Service 97d2fb
  struct Dwarf_CU *cu = attr->cu;
Packit Service 97d2fb
Packit Service 97d2fb
  Dwarf_Off offset;
Packit Service 97d2fb
  if (attr->form == DW_FORM_ref_addr || attr->form == DW_FORM_GNU_ref_alt
Packit Service 97d2fb
      || attr->form == DW_FORM_ref_sup4 || attr->form == DW_FORM_ref_sup8)
Packit Service 97d2fb
    {
Packit Service 97d2fb
      /* This has an absolute offset.  */
Packit Service 97d2fb
Packit Service 97d2fb
      uint8_t ref_size;
Packit Service 97d2fb
      if (cu->version == 2 && attr->form == DW_FORM_ref_addr)
Packit Service 97d2fb
	ref_size = cu->address_size;
Packit Service 97d2fb
      else if (attr->form == DW_FORM_ref_sup4)
Packit Service 97d2fb
	ref_size = 4;
Packit Service 97d2fb
      else if (attr->form == DW_FORM_ref_sup8)
Packit Service 97d2fb
	ref_size = 8;
Packit Service 97d2fb
      else
Packit Service 97d2fb
	ref_size = cu->offset_size;
Packit Service 97d2fb
Packit Service 97d2fb
      Dwarf *dbg_ret = (attr->form == DW_FORM_GNU_ref_alt
Packit Service 97d2fb
			? INTUSE(dwarf_getalt) (cu->dbg) : cu->dbg);
Packit Service 97d2fb
Packit Service 97d2fb
      if (dbg_ret == NULL)
Packit Service 97d2fb
	{
Packit Service 97d2fb
	  __libdw_seterrno (DWARF_E_NO_ALT_DEBUGLINK);
Packit Service 97d2fb
	  return NULL;
Packit Service 97d2fb
	}
Packit Service 97d2fb
Packit Service 97d2fb
      if (__libdw_read_offset (cu->dbg, dbg_ret, IDX_debug_info, attr->valp,
Packit Service 97d2fb
			       ref_size, &offset, IDX_debug_info, 0))
Packit Service 97d2fb
	return NULL;
Packit Service 97d2fb
Packit Service 97d2fb
      return INTUSE(dwarf_offdie) (dbg_ret, offset, result);
Packit Service 97d2fb
    }
Packit Service 97d2fb
Packit Service 97d2fb
  const unsigned char *datap;
Packit Service 97d2fb
  size_t size;
Packit Service 97d2fb
  if (attr->form == DW_FORM_ref_sig8)
Packit Service 97d2fb
    {
Packit Service 97d2fb
      /* This doesn't have an offset, but instead a value we
Packit Service 97d2fb
	 have to match in the type unit headers.  */
Packit Service 97d2fb
Packit Service 97d2fb
      uint64_t sig = read_8ubyte_unaligned (cu->dbg, attr->valp);
Packit Service 97d2fb
      cu = Dwarf_Sig8_Hash_find (&cu->dbg->sig8_hash, sig);
Packit Service 97d2fb
      if (cu == NULL)
Packit Service 97d2fb
	{
Packit Service 97d2fb
	  /* Not seen before.  We have to scan through the type units.
Packit Service 97d2fb
	     Since DWARFv5 these can (also) be found in .debug_info,
Packit Service 97d2fb
	     so scan that first.  */
Packit Service 97d2fb
	  bool scan_debug_types = false;
Packit Service 97d2fb
	  do
Packit Service 97d2fb
	    {
Packit Service 97d2fb
	      cu = __libdw_intern_next_unit (attr->cu->dbg, scan_debug_types);
Packit Service 97d2fb
	      if (cu == NULL)
Packit Service 97d2fb
		{
Packit Service 97d2fb
		  if (scan_debug_types == false)
Packit Service 97d2fb
		    scan_debug_types = true;
Packit Service 97d2fb
		  else
Packit Service 97d2fb
		    {
Packit Service 97d2fb
		      __libdw_seterrno (INTUSE(dwarf_errno) ()
Packit Service 97d2fb
					?: DWARF_E_INVALID_REFERENCE);
Packit Service 97d2fb
		      return NULL;
Packit Service 97d2fb
		    }
Packit Service 97d2fb
		}
Packit Service 97d2fb
	    }
Packit Service 97d2fb
	  while (cu == NULL || cu->unit_id8 != sig);
Packit Service 97d2fb
	}
Packit Service 97d2fb
Packit Service 97d2fb
      int secid = cu_sec_idx (cu);
Packit Service 97d2fb
      datap = cu->dbg->sectiondata[secid]->d_buf;
Packit Service 97d2fb
      size = cu->dbg->sectiondata[secid]->d_size;
Packit Service 97d2fb
      offset = cu->start + cu->subdie_offset;
Packit Service 97d2fb
    }
Packit Service 97d2fb
  else
Packit Service 97d2fb
    {
Packit Service 97d2fb
      /* Other forms produce an offset from the CU.  */
Packit Service 97d2fb
      if (unlikely (__libdw_formref (attr, &offset) != 0))
Packit Service 97d2fb
	return NULL;
Packit Service 97d2fb
Packit Service 97d2fb
      datap = cu->startp;
Packit Service 97d2fb
      size = cu->endp - cu->startp;
Packit Service 97d2fb
    }
Packit Service 97d2fb
Packit Service 97d2fb
  if (unlikely (offset >= size))
Packit Service 97d2fb
    {
Packit Service 97d2fb
      __libdw_seterrno (DWARF_E_INVALID_DWARF);
Packit Service 97d2fb
      return NULL;
Packit Service 97d2fb
    }
Packit Service 97d2fb
Packit Service 97d2fb
  memset (result, '\0', sizeof (Dwarf_Die));
Packit Service 97d2fb
  result->addr = (char *) datap + offset;
Packit Service 97d2fb
  result->cu = cu;
Packit Service 97d2fb
  return result;
Packit Service 97d2fb
}
Packit Service 97d2fb
INTDEF (dwarf_formref_die)