Blame tests/dwarfcfi.c

Packit Service 97d2fb
/* Test program for DWARF (.debug_frame) CFI handling.
Packit Service 97d2fb
   Copyright (C) 2009-2010, 2013, 2015, 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 the GNU General Public License as published by
Packit Service 97d2fb
   the Free Software Foundation; either version 3 of the License, or
Packit Service 97d2fb
   (at your option) any later version.
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
Packit Service 97d2fb
   GNU General Public License for more details.
Packit Service 97d2fb
Packit Service 97d2fb
   You should have received a copy of the GNU General Public License
Packit Service 97d2fb
   along with this program.  If not, see <http://www.gnu.org/licenses/>.  */
Packit Service 97d2fb
Packit Service 97d2fb
#include <config.h>
Packit Service 97d2fb
#include <assert.h>
Packit Service 97d2fb
#include <inttypes.h>
Packit Service 97d2fb
#include ELFUTILS_HEADER(dw)
Packit Service 97d2fb
#include <dwarf.h>
Packit Service 97d2fb
#include <argp.h>
Packit Service 97d2fb
#include <stdio.h>
Packit Service 97d2fb
#include <sys/types.h>
Packit Service 97d2fb
#include <sys/stat.h>
Packit Service 97d2fb
#include <fcntl.h>
Packit Service 97d2fb
#include <locale.h>
Packit Service 97d2fb
#include <stdlib.h>
Packit Service 97d2fb
#include <string.h>
Packit Service 97d2fb
Packit Service 97d2fb
#include "system.h"
Packit Service 97d2fb
#include "../libdw/known-dwarf.h"
Packit Service 97d2fb
Packit Service 97d2fb
static const char *
Packit Service 97d2fb
op_name (unsigned int code)
Packit Service 97d2fb
{
Packit Service 97d2fb
  static const char *const known[] =
Packit Service 97d2fb
    {
Packit Service 97d2fb
#define DWARF_ONE_KNOWN_DW_OP(NAME, CODE) [CODE] = #NAME,
Packit Service 97d2fb
      DWARF_ALL_KNOWN_DW_OP
Packit Service 97d2fb
#undef DWARF_ONE_KNOWN_DW_OP
Packit Service 97d2fb
    };
Packit Service 97d2fb
Packit Service 97d2fb
  if (likely (code < sizeof (known) / sizeof (known[0])))
Packit Service 97d2fb
    return known[code];
Packit Service 97d2fb
Packit Service 97d2fb
  return NULL;
Packit Service 97d2fb
}
Packit Service 97d2fb
Packit Service 97d2fb
static void
Packit Service 97d2fb
print_detail (int result, const Dwarf_Op *ops, size_t nops)
Packit Service 97d2fb
{
Packit Service 97d2fb
  if (result < 0)
Packit Service 97d2fb
    printf ("indeterminate (%s)\n", dwarf_errmsg (-1));
Packit Service 97d2fb
  else if (nops == 0)
Packit Service 97d2fb
    printf ("%s\n", ops == NULL ? "same_value" : "undefined");
Packit Service 97d2fb
  else
Packit Service 97d2fb
    {
Packit Service 97d2fb
      printf ("%s expression:", result == 0 ? "location" : "value");
Packit Service 97d2fb
      for (size_t i = 0; i < nops; ++i)
Packit Service 97d2fb
	{
Packit Service 97d2fb
	  printf (" %s", op_name(ops[i].atom));
Packit Service 97d2fb
	  if (ops[i].number2 == 0)
Packit Service 97d2fb
	    {
Packit Service 97d2fb
	      if (ops[i].atom == DW_OP_addr)
Packit Service 97d2fb
		printf ("(%#" PRIx64 ")", ops[i].number);
Packit Service 97d2fb
	      else if (ops[i].number != 0)
Packit Service 97d2fb
		printf ("(%" PRId64 ")", ops[i].number);
Packit Service 97d2fb
	    }
Packit Service 97d2fb
	  else
Packit Service 97d2fb
	    printf ("(%" PRId64 ",%" PRId64 ")",
Packit Service 97d2fb
		    ops[i].number, ops[i].number2);
Packit Service 97d2fb
	}
Packit Service 97d2fb
      puts ("");
Packit Service 97d2fb
    }
Packit Service 97d2fb
}
Packit Service 97d2fb
Packit Service 97d2fb
static int
Packit Service 97d2fb
handle_address (Dwarf_CFI *cfi, GElf_Addr pc)
Packit Service 97d2fb
{
Packit Service 97d2fb
  Dwarf_Frame *frame;
Packit Service 97d2fb
  int result = dwarf_cfi_addrframe (cfi, pc, &frame);
Packit Service 97d2fb
  if (result != 0)
Packit Service 97d2fb
    {
Packit Service 97d2fb
      printf ("dwarf_cfi_addrframe: %s\n", dwarf_errmsg (-1));
Packit Service 97d2fb
      return 1;
Packit Service 97d2fb
    }
Packit Service 97d2fb
Packit Service 97d2fb
  Dwarf_Addr start = pc;
Packit Service 97d2fb
  Dwarf_Addr end = pc;
Packit Service 97d2fb
  bool signalp;
Packit Service 97d2fb
  int ra_regno = dwarf_frame_info (frame, &start, &end, &signalp);
Packit Service 97d2fb
Packit Service 97d2fb
  printf ("%#" PRIx64 " => [%#" PRIx64 ", %#" PRIx64 "):\n",
Packit Service 97d2fb
	  pc, start, end);
Packit Service 97d2fb
Packit Service 97d2fb
  if (ra_regno < 0)
Packit Service 97d2fb
    printf ("\treturn address register unavailable (%s)\n",
Packit Service 97d2fb
	    dwarf_errmsg (-1));
Packit Service 97d2fb
  else
Packit Service 97d2fb
    printf ("\treturn address in reg%u%s\n",
Packit Service 97d2fb
	    ra_regno, signalp ? " (signal frame)" : "");
Packit Service 97d2fb
Packit Service 97d2fb
  // Point cfa_ops to dummy to match print_detail expectations.
Packit Service 97d2fb
  // (nops == 0 && cfa_ops != NULL => "undefined")
Packit Service 97d2fb
  Dwarf_Op dummy;
Packit Service 97d2fb
  Dwarf_Op *cfa_ops = &dummy;
Packit Service 97d2fb
  size_t cfa_nops;
Packit Service 97d2fb
  result = dwarf_frame_cfa (frame, &cfa_ops, &cfa_nops);
Packit Service 97d2fb
Packit Service 97d2fb
  printf ("\tCFA ");
Packit Service 97d2fb
  print_detail (result, cfa_ops, cfa_nops);
Packit Service 97d2fb
Packit Service 97d2fb
  // Print the location of the first 10 (DWARF nr) registers
Packit Service 97d2fb
  for (int r = 0; r < 10; r++)
Packit Service 97d2fb
    {
Packit Service 97d2fb
      Dwarf_Op ops_mem[3];
Packit Service 97d2fb
      Dwarf_Op *ops;
Packit Service 97d2fb
      size_t nops;
Packit Service 97d2fb
      printf ("\treg%d: ", r);
Packit Service 97d2fb
      int reg_result = dwarf_frame_register (frame, r, ops_mem, &ops, &nops;;
Packit Service 97d2fb
      print_detail (reg_result, ops, nops);
Packit Service 97d2fb
      result |= reg_result;
Packit Service 97d2fb
    }
Packit Service 97d2fb
Packit Service 97d2fb
  free (frame);
Packit Service 97d2fb
  return result;
Packit Service 97d2fb
}
Packit Service 97d2fb
Packit Service 97d2fb
int
Packit Service 97d2fb
main (int argc, char *argv[])
Packit Service 97d2fb
{
Packit Service 97d2fb
  if (argc <= 2)
Packit Service 97d2fb
    error (EXIT_FAILURE, 0, "need file name argument and addresses");
Packit Service 97d2fb
Packit Service 97d2fb
  int fd = open (argv[1], O_RDONLY);
Packit Service 97d2fb
  if (fd == -1)
Packit Service 97d2fb
    error (EXIT_FAILURE, errno, "cannot open input file `%s'", argv[1]);
Packit Service 97d2fb
Packit Service 97d2fb
  elf_version (EV_CURRENT);
Packit Service 97d2fb
Packit Service 97d2fb
  Elf *elf = elf_begin (fd, ELF_C_READ, NULL);
Packit Service 97d2fb
  if (elf == NULL)
Packit Service 97d2fb
    error (EXIT_FAILURE, 0, "cannot create ELF descriptor: %s",
Packit Service 97d2fb
	   elf_errmsg (-1));
Packit Service 97d2fb
Packit Service 97d2fb
  Dwarf *dwarf = dwarf_begin_elf (elf, DWARF_C_READ, NULL);
Packit Service 97d2fb
  if (dwarf == NULL)
Packit Service 97d2fb
    error (EXIT_FAILURE, 0, "cannot create DWARF descriptor: %s",
Packit Service 97d2fb
	   dwarf_errmsg (-1));
Packit Service 97d2fb
Packit Service 97d2fb
  Dwarf_CFI *cfi = dwarf_getcfi (dwarf);
Packit Service 97d2fb
  if (cfi == NULL)
Packit Service 97d2fb
    error (EXIT_FAILURE, 0, "cannot get DWARF CFI from .dwarf_frame: %s",
Packit Service 97d2fb
	   dwarf_errmsg (-1));
Packit Service 97d2fb
Packit Service 97d2fb
  int result = 0;
Packit Service 97d2fb
  int args = 2;
Packit Service 97d2fb
  do
Packit Service 97d2fb
    {
Packit Service 97d2fb
      char *endp;
Packit Service 97d2fb
      uintmax_t addr = strtoumax (argv[args], &endp, 0);
Packit Service 97d2fb
      if (endp != argv[args])
Packit Service 97d2fb
	result |= handle_address (cfi, addr);
Packit Service 97d2fb
      else
Packit Service 97d2fb
	result = 1;
Packit Service 97d2fb
    }
Packit Service 97d2fb
  while (args++ < argc - 1);
Packit Service 97d2fb
Packit Service 97d2fb
  dwarf_end (dwarf);
Packit Service 97d2fb
  elf_end (elf);
Packit Service 97d2fb
Packit Service 97d2fb
  return result;
Packit Service 97d2fb
}