Blame libdw/dwarf_getlocation_attr.c

Packit Service 97d2fb
/* Return DWARF attribute associated with a location expression op.
Packit Service 97d2fb
   Copyright (C) 2013, 2014, 2017, 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 <dwarf.h>
Packit Service 97d2fb
#include <libdwP.h>
Packit Service 97d2fb
Packit Service 97d2fb
static Dwarf_CU *
Packit Service 97d2fb
attr_form_cu (Dwarf_Attribute *attr)
Packit Service 97d2fb
{
Packit Service 97d2fb
  /* If the attribute has block/expr form the data comes from the
Packit Service 97d2fb
     .debug_info from the same cu as the attr.  Otherwise it comes from
Packit Service 97d2fb
     the .debug_loc or .debug_loclists data section.  */
Packit Service 97d2fb
  switch (attr->form)
Packit Service 97d2fb
    {
Packit Service 97d2fb
    case DW_FORM_block1:
Packit Service 97d2fb
    case DW_FORM_block2:
Packit Service 97d2fb
    case DW_FORM_block4:
Packit Service 97d2fb
    case DW_FORM_block:
Packit Service 97d2fb
    case DW_FORM_exprloc:
Packit Service 97d2fb
      return attr->cu;
Packit Service 97d2fb
    default:
Packit Service 97d2fb
      return (attr->cu->version < 5
Packit Service 97d2fb
	      ? attr->cu->dbg->fake_loc_cu
Packit Service 97d2fb
	      : attr->cu->dbg->fake_loclists_cu);
Packit Service 97d2fb
    }
Packit Service 97d2fb
}
Packit Service 97d2fb
Packit Service 97d2fb
static unsigned char *
Packit Service 97d2fb
addr_valp (Dwarf_CU *cu, Dwarf_Word index)
Packit Service 97d2fb
{
Packit Service 97d2fb
  Elf_Data *debug_addr = cu->dbg->sectiondata[IDX_debug_addr];
Packit Service 97d2fb
  if (debug_addr == NULL)
Packit Service 97d2fb
    {
Packit Service 97d2fb
      __libdw_seterrno (DWARF_E_NO_DEBUG_ADDR);
Packit Service 97d2fb
      return NULL;
Packit Service 97d2fb
    }
Packit Service 97d2fb
Packit Service 97d2fb
  Dwarf_Word offset = __libdw_cu_addr_base (cu) + (index * cu->address_size);
Packit Service 97d2fb
  return (unsigned char *) debug_addr->d_buf + offset;
Packit Service 97d2fb
}
Packit Service 97d2fb
Packit Service 97d2fb
int
Packit Service 97d2fb
dwarf_getlocation_attr (Dwarf_Attribute *attr, const Dwarf_Op *op, Dwarf_Attribute *result)
Packit Service 97d2fb
{
Packit Service 97d2fb
  if (attr == NULL)
Packit Service 97d2fb
    return -1;
Packit Service 97d2fb
Packit Service 97d2fb
  switch (op->atom)
Packit Service 97d2fb
    {
Packit Service 97d2fb
      case DW_OP_implicit_value:
Packit Service 97d2fb
	result->code = DW_AT_const_value;
Packit Service 97d2fb
	result->form = DW_FORM_block;
Packit Service 97d2fb
	result->valp = (unsigned char *) (uintptr_t) op->number2;
Packit Service 97d2fb
	result->cu = attr_form_cu (attr);
Packit Service 97d2fb
	break;
Packit Service 97d2fb
Packit Service 97d2fb
      case DW_OP_entry_value:
Packit Service 97d2fb
      case DW_OP_GNU_entry_value:
Packit Service 97d2fb
	result->code = DW_AT_location;
Packit Service 97d2fb
	result->form = DW_FORM_exprloc;
Packit Service 97d2fb
	result->valp = (unsigned char *) (uintptr_t) op->number2;
Packit Service 97d2fb
	result->cu = attr_form_cu (attr);
Packit Service 97d2fb
	break;
Packit Service 97d2fb
Packit Service 97d2fb
      case DW_OP_const_type:
Packit Service 97d2fb
      case DW_OP_GNU_const_type:
Packit Service 97d2fb
	result->code = DW_AT_const_value;
Packit Service 97d2fb
	result->form = DW_FORM_block1;
Packit Service 97d2fb
	result->valp = (unsigned char *) (uintptr_t) op->number2;
Packit Service 97d2fb
	result->cu = attr_form_cu (attr);
Packit Service 97d2fb
	break;
Packit Service 97d2fb
Packit Service 97d2fb
      case DW_OP_GNU_const_index:
Packit Service 97d2fb
      case DW_OP_constx:
Packit Service 97d2fb
	result->code = DW_AT_const_value;
Packit Service 97d2fb
	if (attr->cu->address_size == 4)
Packit Service 97d2fb
	  result->form = DW_FORM_data4;
Packit Service 97d2fb
	else
Packit Service 97d2fb
	  result->form = DW_FORM_data8;
Packit Service 97d2fb
	result->valp = addr_valp (attr->cu, op->number);
Packit Service 97d2fb
	if (result->valp == NULL)
Packit Service 97d2fb
	  return -1;
Packit Service 97d2fb
	result->cu = attr->cu->dbg->fake_addr_cu;
Packit Service 97d2fb
	break;
Packit Service 97d2fb
Packit Service 97d2fb
      case DW_OP_GNU_addr_index:
Packit Service 97d2fb
      case DW_OP_addrx:
Packit Service 97d2fb
	result->code = DW_AT_low_pc;
Packit Service 97d2fb
	result->form = DW_FORM_addr;
Packit Service 97d2fb
	result->valp = addr_valp (attr->cu, op->number);
Packit Service 97d2fb
	if (result->valp == NULL)
Packit Service 97d2fb
	  return -1;
Packit Service 97d2fb
	result->cu = attr->cu->dbg->fake_addr_cu;
Packit Service 97d2fb
	break;
Packit Service 97d2fb
Packit Service 97d2fb
      case DW_OP_call2:
Packit Service 97d2fb
      case DW_OP_call4:
Packit Service 97d2fb
      case DW_OP_call_ref:
Packit Service 97d2fb
	{
Packit Service 97d2fb
	  Dwarf_Die die;
Packit Service 97d2fb
	  if (INTUSE(dwarf_getlocation_die) (attr, op, &die) != 0)
Packit Service 97d2fb
	    return -1;
Packit Service 97d2fb
	  if (INTUSE(dwarf_attr) (&die, DW_AT_location, result) == NULL)
Packit Service 97d2fb
	    {
Packit Service 97d2fb
	      __libdw_empty_loc_attr (result);
Packit Service 97d2fb
	      return 0;
Packit Service 97d2fb
	    }
Packit Service 97d2fb
	}
Packit Service 97d2fb
	break;
Packit Service 97d2fb
Packit Service 97d2fb
      case DW_OP_implicit_pointer:
Packit Service 97d2fb
      case DW_OP_GNU_implicit_pointer:
Packit Service 97d2fb
      case DW_OP_GNU_variable_value:
Packit Service 97d2fb
	{
Packit Service 97d2fb
	  Dwarf_Die die;
Packit Service 97d2fb
	  if (INTUSE(dwarf_getlocation_die) (attr, op, &die) != 0)
Packit Service 97d2fb
	    return -1;
Packit Service 97d2fb
	  if (INTUSE(dwarf_attr) (&die, DW_AT_location, result) == NULL
Packit Service 97d2fb
	      && INTUSE(dwarf_attr) (&die, DW_AT_const_value, result) == NULL)
Packit Service 97d2fb
	    {
Packit Service 97d2fb
	      __libdw_empty_loc_attr (result);
Packit Service 97d2fb
	      return 0;
Packit Service 97d2fb
	    }
Packit Service 97d2fb
	}
Packit Service 97d2fb
	break;
Packit Service 97d2fb
Packit Service 97d2fb
      default:
Packit Service 97d2fb
	__libdw_seterrno (DWARF_E_INVALID_ACCESS);
Packit Service 97d2fb
	return -1;
Packit Service 97d2fb
    }
Packit Service 97d2fb
Packit Service 97d2fb
  return 0;
Packit Service 97d2fb
}