Blame tests/addrscopes.c

Packit Service 97d2fb
/* Test program for dwarf_getscopes.
Packit Service 97d2fb
   Copyright (C) 2005, 2014 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(dwfl)
Packit Service 97d2fb
#include <dwarf.h>
Packit Service 97d2fb
#include <argp.h>
Packit Service 97d2fb
#include <stdio.h>
Packit Service 97d2fb
#include <stdio_ext.h>
Packit Service 97d2fb
#include <locale.h>
Packit Service 97d2fb
#include <stdlib.h>
Packit Service 97d2fb
#include <string.h>
Packit Service 97d2fb
#include "system.h"
Packit Service 97d2fb
Packit Service 97d2fb
Packit Service 97d2fb
static void
Packit Service 97d2fb
paddr (const char *prefix, Dwarf_Addr addr, Dwfl_Line *line)
Packit Service 97d2fb
{
Packit Service 97d2fb
  const char *src;
Packit Service 97d2fb
  int lineno, linecol;
Packit Service 97d2fb
  if (line != NULL
Packit Service 97d2fb
      && (src = dwfl_lineinfo (line, &addr, &lineno, &linecol,
Packit Service 97d2fb
			       NULL, NULL)) != NULL)
Packit Service 97d2fb
    {
Packit Service 97d2fb
      if (linecol != 0)
Packit Service 97d2fb
	printf ("%s%#" PRIx64 " (%s:%d:%d)",
Packit Service 97d2fb
		prefix, addr, src, lineno, linecol);
Packit Service 97d2fb
      else
Packit Service 97d2fb
	printf ("%s%#" PRIx64 " (%s:%d)",
Packit Service 97d2fb
		prefix, addr, src, lineno);
Packit Service 97d2fb
    }
Packit Service 97d2fb
  else
Packit Service 97d2fb
    printf ("%s%#" PRIx64, prefix, addr);
Packit Service 97d2fb
}
Packit Service 97d2fb
Packit Service 97d2fb
static void
Packit Service 97d2fb
print_vars (unsigned int indent, Dwarf_Die *die)
Packit Service 97d2fb
{
Packit Service 97d2fb
  Dwarf_Die child;
Packit Service 97d2fb
  if (dwarf_child (die, &child) == 0)
Packit Service 97d2fb
    do
Packit Service 97d2fb
      switch (dwarf_tag (&child))
Packit Service 97d2fb
	{
Packit Service 97d2fb
	case DW_TAG_variable:
Packit Service 97d2fb
	case DW_TAG_formal_parameter:
Packit Service 97d2fb
	  printf ("%*s%-30s[%6" PRIx64 "]\n", indent, "",
Packit Service 97d2fb
		  dwarf_diename (&child),
Packit Service 97d2fb
		  (uint64_t) dwarf_dieoffset (&child));
Packit Service 97d2fb
	  break;
Packit Service 97d2fb
	default:
Packit Service 97d2fb
	  break;
Packit Service 97d2fb
	}
Packit Service 97d2fb
    while (dwarf_siblingof (&child, &child) == 0);
Packit Service 97d2fb
Packit Service 97d2fb
  Dwarf_Attribute attr_mem;
Packit Service 97d2fb
  Dwarf_Die origin;
Packit Service 97d2fb
  if (dwarf_hasattr (die, DW_AT_abstract_origin)
Packit Service 97d2fb
      && dwarf_formref_die (dwarf_attr (die, DW_AT_abstract_origin, &attr_mem),
Packit Service 97d2fb
			    &origin) != NULL
Packit Service 97d2fb
      && dwarf_child (&origin, &child) == 0)
Packit Service 97d2fb
    do
Packit Service 97d2fb
      switch (dwarf_tag (&child))
Packit Service 97d2fb
	{
Packit Service 97d2fb
	case DW_TAG_variable:
Packit Service 97d2fb
	case DW_TAG_formal_parameter:
Packit Service 97d2fb
	  printf ("%*s%s (abstract)\n", indent, "",
Packit Service 97d2fb
		  dwarf_diename (&child));
Packit Service 97d2fb
	  break;
Packit Service 97d2fb
	default:
Packit Service 97d2fb
	  break;
Packit Service 97d2fb
	}
Packit Service 97d2fb
    while (dwarf_siblingof (&child, &child) == 0);
Packit Service 97d2fb
}
Packit Service 97d2fb
Packit Service 97d2fb
Packit Service 97d2fb
#define INDENT 4
Packit Service 97d2fb
Packit Service 97d2fb
static void
Packit Service 97d2fb
handle_address (GElf_Addr pc, Dwfl *dwfl)
Packit Service 97d2fb
{
Packit Service 97d2fb
  Dwarf_Addr cubias;
Packit Service 97d2fb
  Dwarf_Die *cudie = dwfl_addrdie (dwfl, pc, &cubias);
Packit Service 97d2fb
  if (cudie == NULL)
Packit Service 97d2fb
    error (EXIT_FAILURE, 0, "dwfl_addrdie: %s", dwfl_errmsg (-1));
Packit Service 97d2fb
Packit Service 97d2fb
  Dwarf_Die *scopes;
Packit Service 97d2fb
  int n = dwarf_getscopes (cudie, pc - cubias, &scopes);
Packit Service 97d2fb
  if (n < 0)
Packit Service 97d2fb
    error (EXIT_FAILURE, 0, "dwarf_getscopes: %s", dwarf_errmsg (-1));
Packit Service 97d2fb
  else if (n == 0)
Packit Service 97d2fb
    printf ("%#" PRIx64 ": not in any scope\n", pc);
Packit Service 97d2fb
  else
Packit Service 97d2fb
    {
Packit Service 97d2fb
      printf ("%#" PRIx64 ":\n", pc);
Packit Service 97d2fb
      unsigned int indent = 0;
Packit Service 97d2fb
      while (n-- > 0)
Packit Service 97d2fb
	{
Packit Service 97d2fb
	  Dwarf_Die *const die = &scopes[n];
Packit Service 97d2fb
Packit Service 97d2fb
	  indent += INDENT;
Packit Service 97d2fb
	  printf ("%*s%s (%#x)", indent, "",
Packit Service 97d2fb
		  dwarf_diename (die) ?: "<unnamed>",
Packit Service 97d2fb
		  dwarf_tag (die));
Packit Service 97d2fb
Packit Service 97d2fb
	  Dwarf_Addr lowpc, highpc;
Packit Service 97d2fb
	  if (dwarf_lowpc (die, &lowpc) == 0
Packit Service 97d2fb
	      && dwarf_highpc (die, &highpc) == 0)
Packit Service 97d2fb
	    {
Packit Service 97d2fb
	      lowpc += cubias;
Packit Service 97d2fb
	      highpc += cubias;
Packit Service 97d2fb
	      Dwfl_Line *loline = dwfl_getsrc (dwfl, lowpc);
Packit Service 97d2fb
	      Dwfl_Line *hiline = dwfl_getsrc (dwfl, highpc - 1);
Packit Service 97d2fb
	      paddr (": ", lowpc, loline);
Packit Service 97d2fb
	      if (highpc != lowpc)
Packit Service 97d2fb
		paddr (" .. ", highpc - 1, hiline == loline ? NULL : hiline);
Packit Service 97d2fb
	    }
Packit Service 97d2fb
	  puts ("");
Packit Service 97d2fb
Packit Service 97d2fb
	  print_vars (indent + INDENT, die);
Packit Service 97d2fb
	}
Packit Service 97d2fb
      free (scopes);
Packit Service 97d2fb
    }
Packit Service 97d2fb
}
Packit Service 97d2fb
Packit Service 97d2fb
int
Packit Service 97d2fb
main (int argc, char *argv[])
Packit Service 97d2fb
{
Packit Service 97d2fb
  int remaining;
Packit Service 97d2fb
Packit Service 97d2fb
  /* Set locale.  */
Packit Service 97d2fb
  (void) setlocale (LC_ALL, "");
Packit Service 97d2fb
Packit Service 97d2fb
  Dwfl *dwfl = NULL;
Packit Service 97d2fb
  (void) argp_parse (dwfl_standard_argp (), argc, argv, 0, &remaining, &dwfl);
Packit Service 97d2fb
  assert (dwfl != NULL);
Packit Service 97d2fb
Packit Service 97d2fb
  int result = 0;
Packit Service 97d2fb
Packit Service 97d2fb
  /* Now handle the addresses.  In case none are given on the command
Packit Service 97d2fb
     line, read from stdin.  */
Packit Service 97d2fb
  if (remaining == argc)
Packit Service 97d2fb
    {
Packit Service 97d2fb
      /* We use no threads here which can interfere with handling a stream.  */
Packit Service 97d2fb
      (void) __fsetlocking (stdin, FSETLOCKING_BYCALLER);
Packit Service 97d2fb
Packit Service 97d2fb
      char *buf = NULL;
Packit Service 97d2fb
      size_t len = 0;
Packit Service 97d2fb
      while (!feof_unlocked (stdin))
Packit Service 97d2fb
	{
Packit Service 97d2fb
	  if (getline (&buf, &len, stdin) < 0)
Packit Service 97d2fb
	    break;
Packit Service 97d2fb
Packit Service 97d2fb
	  char *endp;
Packit Service 97d2fb
	  uintmax_t addr = strtoumax (buf, &endp, 0);
Packit Service 97d2fb
	  if (endp != buf)
Packit Service 97d2fb
	    handle_address (addr, dwfl);
Packit Service 97d2fb
	  else
Packit Service 97d2fb
	    result = 1;
Packit Service 97d2fb
	}
Packit Service 97d2fb
Packit Service 97d2fb
      free (buf);
Packit Service 97d2fb
    }
Packit Service 97d2fb
  else
Packit Service 97d2fb
    {
Packit Service 97d2fb
      do
Packit Service 97d2fb
	{
Packit Service 97d2fb
	  char *endp;
Packit Service 97d2fb
	  uintmax_t addr = strtoumax (argv[remaining], &endp, 0);
Packit Service 97d2fb
	  if (endp != argv[remaining])
Packit Service 97d2fb
	    handle_address (addr, dwfl);
Packit Service 97d2fb
	  else
Packit Service 97d2fb
	    result = 1;
Packit Service 97d2fb
	}
Packit Service 97d2fb
      while (++remaining < argc);
Packit Service 97d2fb
    }
Packit Service 97d2fb
Packit Service 97d2fb
  dwfl_end (dwfl);
Packit Service 97d2fb
Packit Service 97d2fb
  return result;
Packit Service 97d2fb
}