Blame tests/dwarfcfi.c

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