Blame libdw/encoded-value.h

Packit Service 97d2fb
/* DW_EH_PE_* support for libdw unwinder.
Packit Service 97d2fb
   Copyright (C) 2009-2010, 2014, 2015 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
#ifndef _ENCODED_VALUE_H
Packit Service 97d2fb
#define _ENCODED_VALUE_H 1
Packit Service 97d2fb
Packit Service 97d2fb
#include <dwarf.h>
Packit Service 97d2fb
#include <stdlib.h>
Packit Service 97d2fb
#include "libdwP.h"
Packit Service 97d2fb
#include "../libelf/common.h"
Packit Service 97d2fb
Packit Service 97d2fb
Packit Service 97d2fb
/* Returns zero if the value is omitted, the encoding is unknown or
Packit Service 97d2fb
   the (leb128) size cannot be determined.  */
Packit Service 97d2fb
static size_t __attribute__ ((unused))
Packit Service 97d2fb
encoded_value_size (const Elf_Data *data, const unsigned char e_ident[],
Packit Service 97d2fb
		    uint8_t encoding, const uint8_t *p)
Packit Service 97d2fb
{
Packit Service 97d2fb
  if (encoding == DW_EH_PE_omit)
Packit Service 97d2fb
    return 0;
Packit Service 97d2fb
Packit Service 97d2fb
  switch (encoding & 0x07)
Packit Service 97d2fb
    {
Packit Service 97d2fb
    case DW_EH_PE_udata2:
Packit Service 97d2fb
      return 2;
Packit Service 97d2fb
    case DW_EH_PE_udata4:
Packit Service 97d2fb
      return 4;
Packit Service 97d2fb
    case DW_EH_PE_udata8:
Packit Service 97d2fb
      return 8;
Packit Service 97d2fb
Packit Service 97d2fb
    case DW_EH_PE_absptr:
Packit Service 97d2fb
      return e_ident[EI_CLASS] == ELFCLASS32 ? 4 : 8;
Packit Service 97d2fb
Packit Service 97d2fb
    case DW_EH_PE_uleb128:
Packit Service 97d2fb
      if (p != NULL)
Packit Service 97d2fb
	{
Packit Service 97d2fb
	  const uint8_t *end = p;
Packit Service 97d2fb
	  while (end < (uint8_t *) data->d_buf + data->d_size)
Packit Service 97d2fb
	    if (*end++ & 0x80u)
Packit Service 97d2fb
	      return end - p;
Packit Service 97d2fb
	}
Packit Service 97d2fb
      return 0;
Packit Service 97d2fb
Packit Service 97d2fb
    default:
Packit Service 97d2fb
      return 0;
Packit Service 97d2fb
    }
Packit Service 97d2fb
}
Packit Service 97d2fb
Packit Service 97d2fb
/* Returns zero when value was read successfully, minus one otherwise.  */
Packit Service 97d2fb
static inline int __attribute__ ((unused))
Packit Service 97d2fb
__libdw_cfi_read_address_inc (const Dwarf_CFI *cache,
Packit Service 97d2fb
			      const unsigned char **addrp,
Packit Service 97d2fb
			      int width, Dwarf_Addr *ret)
Packit Service 97d2fb
{
Packit Service 97d2fb
  width = width ?: cache->e_ident[EI_CLASS] == ELFCLASS32 ? 4 : 8;
Packit Service 97d2fb
Packit Service 97d2fb
  if (cache->dbg != NULL)
Packit Service 97d2fb
    return __libdw_read_address_inc (cache->dbg, IDX_debug_frame,
Packit Service 97d2fb
				     addrp, width, ret);
Packit Service 97d2fb
Packit Service 97d2fb
  /* Only .debug_frame might have relocation to consider.
Packit Service 97d2fb
     Read plain values from .eh_frame data.  */
Packit Service 97d2fb
Packit Service 97d2fb
  const unsigned char *endp = cache->data->d.d_buf + cache->data->d.d_size;
Packit Service 97d2fb
  Dwarf eh_dbg = { .other_byte_order = MY_ELFDATA != cache->e_ident[EI_DATA] };
Packit Service 97d2fb
Packit Service 97d2fb
  if (width == 4)
Packit Service 97d2fb
    {
Packit Service 97d2fb
      if (unlikely (*addrp + 4 > endp))
Packit Service 97d2fb
	{
Packit Service 97d2fb
	invalid_data:
Packit Service 97d2fb
	  __libdw_seterrno (DWARF_E_INVALID_CFI);
Packit Service 97d2fb
	  return -1;
Packit Service 97d2fb
	}
Packit Service 97d2fb
      *ret = read_4ubyte_unaligned_inc (&eh_dbg, *addrp);
Packit Service 97d2fb
    }
Packit Service 97d2fb
  else
Packit Service 97d2fb
    {
Packit Service 97d2fb
      if (unlikely (*addrp + 8 > endp))
Packit Service 97d2fb
	goto invalid_data;
Packit Service 97d2fb
      *ret = read_8ubyte_unaligned_inc (&eh_dbg, *addrp);
Packit Service 97d2fb
    }
Packit Service 97d2fb
  return 0;
Packit Service 97d2fb
}
Packit Service 97d2fb
Packit Service 97d2fb
/* Returns true on error, false otherwise. */
Packit Service 97d2fb
static bool __attribute__ ((unused))
Packit Service 97d2fb
read_encoded_value (const Dwarf_CFI *cache, uint8_t encoding,
Packit Service 97d2fb
		    const uint8_t **p, Dwarf_Addr *result)
Packit Service 97d2fb
{
Packit Service 97d2fb
  *result = 0;
Packit Service 97d2fb
  switch (encoding & 0x70)
Packit Service 97d2fb
    {
Packit Service 97d2fb
    case DW_EH_PE_absptr:
Packit Service 97d2fb
      break;
Packit Service 97d2fb
    case DW_EH_PE_pcrel:
Packit Service 97d2fb
      *result = (cache->frame_vaddr
Packit Service 97d2fb
		 + (*p - (const uint8_t *) cache->data->d.d_buf));
Packit Service 97d2fb
      break;
Packit Service 97d2fb
    case DW_EH_PE_textrel:
Packit Service 97d2fb
      // ia64: segrel
Packit Service 97d2fb
      *result = cache->textrel;
Packit Service 97d2fb
      break;
Packit Service 97d2fb
    case DW_EH_PE_datarel:
Packit Service 97d2fb
      // i386: GOTOFF
Packit Service 97d2fb
      // ia64: gprel
Packit Service 97d2fb
      *result = cache->datarel;
Packit Service 97d2fb
      break;
Packit Service 97d2fb
    case DW_EH_PE_funcrel:	/* XXX */
Packit Service 97d2fb
      break;
Packit Service 97d2fb
    case DW_EH_PE_aligned:
Packit Service 97d2fb
      {
Packit Service 97d2fb
	const size_t size = encoded_value_size (&cache->data->d,
Packit Service 97d2fb
						cache->e_ident,
Packit Service 97d2fb
						encoding, *p);
Packit Service 97d2fb
	if (unlikely (size == 0))
Packit Service 97d2fb
	  return true;
Packit Service 97d2fb
	size_t align = ((cache->frame_vaddr
Packit Service 97d2fb
			 + (*p - (const uint8_t *) cache->data->d.d_buf))
Packit Service 97d2fb
			& (size - 1));
Packit Service 97d2fb
	if (align != 0)
Packit Service 97d2fb
	  *p += size - align;
Packit Service 97d2fb
	break;
Packit Service 97d2fb
      }
Packit Service 97d2fb
Packit Service 97d2fb
    default:
Packit Service 97d2fb
      __libdw_seterrno (DWARF_E_INVALID_CFI);
Packit Service 97d2fb
      return true;
Packit Service 97d2fb
    }
Packit Service 97d2fb
Packit Service 97d2fb
  Dwarf_Addr value = 0;
Packit Service 97d2fb
  const unsigned char *endp = cache->data->d.d_buf + cache->data->d.d_size;
Packit Service 97d2fb
  switch (encoding & 0x0f)
Packit Service 97d2fb
    {
Packit Service 97d2fb
    case DW_EH_PE_udata2:
Packit Service 97d2fb
      if (unlikely (*p + 2 > endp))
Packit Service 97d2fb
	{
Packit Service 97d2fb
	invalid_data:
Packit Service 97d2fb
	  __libdw_seterrno (DWARF_E_INVALID_CFI);
Packit Service 97d2fb
	  return true;
Packit Service 97d2fb
	}
Packit Service 97d2fb
      value = read_2ubyte_unaligned_inc (cache, *p);
Packit Service 97d2fb
      break;
Packit Service 97d2fb
Packit Service 97d2fb
    case DW_EH_PE_sdata2:
Packit Service 97d2fb
      if (unlikely (*p + 2 > endp))
Packit Service 97d2fb
	goto invalid_data;
Packit Service 97d2fb
      value = read_2sbyte_unaligned_inc (cache, *p);
Packit Service 97d2fb
      break;
Packit Service 97d2fb
Packit Service 97d2fb
    case DW_EH_PE_udata4:
Packit Service 97d2fb
      if (unlikely (__libdw_cfi_read_address_inc (cache, p, 4, &value) != 0))
Packit Service 97d2fb
	return true;
Packit Service 97d2fb
      break;
Packit Service 97d2fb
Packit Service 97d2fb
    case DW_EH_PE_sdata4:
Packit Service 97d2fb
      if (unlikely (__libdw_cfi_read_address_inc (cache, p, 4, &value) != 0))
Packit Service 97d2fb
	return true;
Packit Service 97d2fb
      value = (Dwarf_Sword) (Elf32_Sword) value; /* Sign-extend.  */
Packit Service 97d2fb
      break;
Packit Service 97d2fb
Packit Service 97d2fb
    case DW_EH_PE_udata8:
Packit Service 97d2fb
    case DW_EH_PE_sdata8:
Packit Service 97d2fb
      if (unlikely (__libdw_cfi_read_address_inc (cache, p, 8, &value) != 0))
Packit Service 97d2fb
	return true;
Packit Service 97d2fb
      break;
Packit Service 97d2fb
Packit Service 97d2fb
    case DW_EH_PE_absptr:
Packit Service 97d2fb
      if (unlikely (__libdw_cfi_read_address_inc (cache, p, 0, &value) != 0))
Packit Service 97d2fb
	return true;
Packit Service 97d2fb
      break;
Packit Service 97d2fb
Packit Service 97d2fb
    case DW_EH_PE_uleb128:
Packit Service 97d2fb
      get_uleb128 (value, *p, endp);
Packit Service 97d2fb
      break;
Packit Service 97d2fb
Packit Service 97d2fb
    case DW_EH_PE_sleb128:
Packit Service 97d2fb
      get_sleb128 (value, *p, endp);
Packit Service 97d2fb
      break;
Packit Service 97d2fb
Packit Service 97d2fb
    default:
Packit Service 97d2fb
      __libdw_seterrno (DWARF_E_INVALID_CFI);
Packit Service 97d2fb
      return true;
Packit Service 97d2fb
    }
Packit Service 97d2fb
Packit Service 97d2fb
  *result += value;
Packit Service 97d2fb
Packit Service 97d2fb
  if (encoding & DW_EH_PE_indirect)
Packit Service 97d2fb
    {
Packit Service 97d2fb
      if (unlikely (*result < cache->frame_vaddr))
Packit Service 97d2fb
	return true;
Packit Service 97d2fb
      *result -= cache->frame_vaddr;
Packit Service 97d2fb
      size_t ptrsize = encoded_value_size (NULL, cache->e_ident,
Packit Service 97d2fb
					   DW_EH_PE_absptr, NULL);
Packit Service 97d2fb
      if (unlikely (cache->data->d.d_size < ptrsize
Packit Service 97d2fb
		    || *result > (cache->data->d.d_size - ptrsize)))
Packit Service 97d2fb
	return true;
Packit Service 97d2fb
      const uint8_t *ptr = cache->data->d.d_buf + *result;
Packit Service 97d2fb
      if (unlikely (__libdw_cfi_read_address_inc (cache, &ptr, 0, result)
Packit Service 97d2fb
		    != 0))
Packit Service 97d2fb
	return true;
Packit Service 97d2fb
    }
Packit Service 97d2fb
Packit Service 97d2fb
  return false;
Packit Service 97d2fb
}
Packit Service 97d2fb
Packit Service 97d2fb
#endif	/* encoded-value.h */