Blame libdw/dwarf_frame_register.c

Packit 032894
/* Get register location expression for frame.
Packit 032894
   Copyright (C) 2009-2010, 2014 Red Hat, Inc.
Packit 032894
   This file is part of elfutils.
Packit 032894
Packit 032894
   This file is free software; you can redistribute it and/or modify
Packit 032894
   it under the terms of either
Packit 032894
Packit 032894
     * the GNU Lesser General Public License as published by the Free
Packit 032894
       Software Foundation; either version 3 of the License, or (at
Packit 032894
       your option) any later version
Packit 032894
Packit 032894
   or
Packit 032894
Packit 032894
     * the GNU General Public License as published by the Free
Packit 032894
       Software Foundation; either version 2 of the License, or (at
Packit 032894
       your option) any later version
Packit 032894
Packit 032894
   or both in parallel, as here.
Packit 032894
Packit 032894
   elfutils is distributed in the hope that it will be useful, but
Packit 032894
   WITHOUT ANY WARRANTY; without even the implied warranty of
Packit 032894
   MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
Packit 032894
   General Public License for more details.
Packit 032894
Packit 032894
   You should have received copies of the GNU General Public License and
Packit 032894
   the GNU Lesser General Public License along with this program.  If
Packit 032894
   not, see <http://www.gnu.org/licenses/>.  */
Packit 032894
Packit 032894
#ifdef HAVE_CONFIG_H
Packit 032894
# include <config.h>
Packit 032894
#endif
Packit 032894
Packit 032894
#include "cfi.h"
Packit 032894
#include <dwarf.h>
Packit 032894
Packit 032894
int
Packit 032894
dwarf_frame_register (Dwarf_Frame *fs, int regno, Dwarf_Op *ops_mem,
Packit 032894
		      Dwarf_Op **ops, size_t *nops)
Packit 032894
{
Packit 032894
  /* Maybe there was a previous error.  */
Packit 032894
  if (fs == NULL)
Packit 032894
    return -1;
Packit 032894
Packit 032894
  if (unlikely (regno < 0))
Packit 032894
    {
Packit 032894
      __libdw_seterrno (DWARF_E_INVALID_ACCESS);
Packit 032894
      return -1;
Packit 032894
    }
Packit 032894
Packit 032894
  *ops = ops_mem;
Packit 032894
  *nops = 0;
Packit 032894
Packit 032894
  if (unlikely ((size_t) regno >= fs->nregs))
Packit 032894
    goto default_rule;
Packit 032894
Packit 032894
  const struct dwarf_frame_register *reg = &fs->regs[regno];
Packit 032894
Packit 032894
  switch (reg->rule)
Packit 032894
    {
Packit 032894
    case reg_unspecified:
Packit 032894
    default_rule:
Packit 032894
      /* Use the default rule for registers not yet mentioned in CFI.  */
Packit 032894
      if (fs->cache->default_same_value)
Packit 032894
	goto same_value;
Packit 032894
      FALLTHROUGH;
Packit 032894
    case reg_undefined:
Packit 032894
      /* The value is known to be unavailable.  */
Packit 032894
      break;
Packit 032894
Packit 032894
    case reg_same_value:
Packit 032894
    same_value:
Packit 032894
      /* The location is not known here, but the caller might know it.  */
Packit 032894
      *ops = NULL;
Packit 032894
      break;
Packit 032894
Packit 032894
    case reg_offset:
Packit 032894
    case reg_val_offset:
Packit 032894
      ops_mem[(*nops)++] = (Dwarf_Op) { .atom = DW_OP_call_frame_cfa };
Packit 032894
      if (reg->value != 0)
Packit 032894
	ops_mem[(*nops)++] = (Dwarf_Op) { .atom = DW_OP_plus_uconst,
Packit 032894
					  .number = reg->value };
Packit 032894
      if (reg->rule == reg_val_offset)
Packit 032894
	/* A value, not a location.  */
Packit 032894
	ops_mem[(*nops)++] = (Dwarf_Op) { .atom = DW_OP_stack_value };
Packit 032894
      *ops = ops_mem;
Packit 032894
      break;
Packit 032894
Packit 032894
    case reg_register:
Packit 032894
      ops_mem[(*nops)++] = (Dwarf_Op) { .atom = DW_OP_regx,
Packit 032894
					.number = reg->value };
Packit 032894
      break;
Packit 032894
Packit 032894
    case reg_val_expression:
Packit 032894
    case reg_expression:
Packit 032894
      {
Packit 032894
	unsigned int address_size = (fs->cache->e_ident[EI_CLASS] == ELFCLASS32
Packit 032894
				     ? 4 : 8);
Packit 032894
Packit 032894
	Dwarf_Block block;
Packit 032894
	const uint8_t *p = fs->cache->data->d.d_buf + reg->value;
Packit 032894
	const uint8_t *end = (fs->cache->data->d.d_buf
Packit 032894
			      + fs->cache->data->d.d_size);
Packit 032894
	get_uleb128 (block.length, p, end);
Packit 032894
	block.data = (void *) p;
Packit 032894
Packit 032894
	/* Parse the expression into internal form.  */
Packit 032894
	if (__libdw_intern_expression (NULL,
Packit 032894
				       fs->cache->other_byte_order,
Packit 032894
				       address_size, 4,
Packit 032894
				       &fs->cache->expr_tree, &block,
Packit 032894
				       true, reg->rule == reg_val_expression,
Packit 032894
				       ops, nops, IDX_debug_frame) < 0)
Packit 032894
	  return -1;
Packit 032894
	break;
Packit 032894
      }
Packit 032894
    }
Packit 032894
Packit 032894
  return 0;
Packit 032894
}