Blame tests/funcscopes.c

Packit 032894
/* Test program for dwarf_getscopes.
Packit 032894
   Copyright (C) 2005, 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 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(dwfl)
Packit 032894
#include <dwarf.h>
Packit 032894
#include <argp.h>
Packit 032894
#include <stdio.h>
Packit 032894
#include <stdio_ext.h>
Packit 032894
#include <locale.h>
Packit 032894
#include <stdlib.h>
Packit 032894
#include <string.h>
Packit 032894
#include <fnmatch.h>
Packit 032894
#include "system.h"
Packit 032894
Packit 032894
Packit 032894
static void
Packit 032894
paddr (const char *prefix, Dwarf_Addr addr, Dwfl_Line *line)
Packit 032894
{
Packit 032894
  const char *src;
Packit 032894
  int lineno, linecol;
Packit 032894
  if (line != NULL
Packit 032894
      && (src = dwfl_lineinfo (line, &addr, &lineno, &linecol,
Packit 032894
			       NULL, NULL)) != NULL)
Packit 032894
    {
Packit 032894
      if (linecol != 0)
Packit 032894
	printf ("%s%#" PRIx64 " (%s:%d:%d)",
Packit 032894
		prefix, addr, src, lineno, linecol);
Packit 032894
      else
Packit 032894
	printf ("%s%#" PRIx64 " (%s:%d)",
Packit 032894
		prefix, addr, src, lineno);
Packit 032894
    }
Packit 032894
  else
Packit 032894
    printf ("%s%#" PRIx64, prefix, addr);
Packit 032894
}
Packit 032894
Packit 032894
Packit 032894
static void
Packit 032894
print_vars (unsigned int indent, Dwarf_Die *die)
Packit 032894
{
Packit 032894
  Dwarf_Die child;
Packit 032894
  if (dwarf_child (die, &child) == 0)
Packit 032894
    do
Packit 032894
      switch (dwarf_tag (&child))
Packit 032894
	{
Packit 032894
	case DW_TAG_variable:
Packit 032894
	case DW_TAG_formal_parameter:
Packit 032894
	  printf ("%*s%-30s[%6" PRIx64 "]\n", indent, "",
Packit 032894
		  dwarf_diename (&child),
Packit 032894
		  (uint64_t) dwarf_dieoffset (&child));
Packit 032894
	  break;
Packit 032894
	default:
Packit 032894
	  break;
Packit 032894
	}
Packit 032894
    while (dwarf_siblingof (&child, &child) == 0);
Packit 032894
Packit 032894
  Dwarf_Attribute attr_mem;
Packit 032894
  Dwarf_Die origin;
Packit 032894
  if (dwarf_hasattr (die, DW_AT_abstract_origin)
Packit 032894
      && dwarf_formref_die (dwarf_attr (die, DW_AT_abstract_origin, &attr_mem),
Packit 032894
			    &origin) != NULL
Packit 032894
      && dwarf_child (&origin, &child) == 0)
Packit 032894
    do
Packit 032894
      switch (dwarf_tag (&child))
Packit 032894
	{
Packit 032894
	case DW_TAG_variable:
Packit 032894
	case DW_TAG_formal_parameter:
Packit 032894
	  printf ("%*s%s (abstract)\n", indent, "",
Packit 032894
		  dwarf_diename (&child));
Packit 032894
	  break;
Packit 032894
	default:
Packit 032894
	  break;
Packit 032894
	}
Packit 032894
    while (dwarf_siblingof (&child, &child) == 0);
Packit 032894
}
Packit 032894
Packit 032894
Packit 032894
#define INDENT 4
Packit 032894
Packit 032894
struct args
Packit 032894
{
Packit 032894
  Dwfl *dwfl;
Packit 032894
  Dwarf_Die *cu;
Packit 032894
  Dwarf_Addr dwbias;
Packit 032894
  char **argv;
Packit 032894
};
Packit 032894
Packit 032894
static int
Packit 032894
handle_function (Dwarf_Die *funcdie, void *arg)
Packit 032894
{
Packit 032894
  struct args *a = arg;
Packit 032894
Packit 032894
  const char *name = dwarf_diename (funcdie);
Packit 032894
  char **argv = a->argv;
Packit 032894
  if (argv[0] != NULL)
Packit 032894
    {
Packit 032894
      bool match;
Packit 032894
      do
Packit 032894
	match = fnmatch (*argv, name, 0) == 0;
Packit 032894
      while (!match && *++argv);
Packit 032894
      if (!match)
Packit 032894
	return 0;
Packit 032894
    }
Packit 032894
Packit 032894
  Dwarf_Die *scopes;
Packit 032894
  int n = dwarf_getscopes_die (funcdie, &scopes);
Packit 032894
  if (n <= 0)
Packit 032894
    error (EXIT_FAILURE, 0, "dwarf_getscopes_die: %s", dwarf_errmsg (-1));
Packit 032894
  else
Packit 032894
    {
Packit 032894
      Dwarf_Addr start, end;
Packit 032894
      const char *fname;
Packit 032894
      const char *modname = dwfl_module_info (dwfl_cumodule (a->cu), NULL,
Packit 032894
					      &start, &end,
Packit 032894
					      NULL, NULL,
Packit 032894
					      &fname, NULL);
Packit 032894
      if (modname == NULL)
Packit 032894
	error (EXIT_FAILURE, 0, "dwfl_module_info: %s", dwarf_errmsg (-1));
Packit 032894
      if (modname[0] == '\0')
Packit 032894
	modname = fname;
Packit 032894
      printf ("%s: %#" PRIx64 " .. %#" PRIx64 "\n", modname, start, end);
Packit 032894
Packit 032894
      unsigned int indent = 0;
Packit 032894
      while (n-- > 0)
Packit 032894
	{
Packit 032894
	  Dwarf_Die *const die = &scopes[n];
Packit 032894
Packit 032894
	  indent += INDENT;
Packit 032894
	  printf ("%*s%s (%#x)", indent, "",
Packit 032894
		  dwarf_diename (die) ?: "<unnamed>",
Packit 032894
		  dwarf_tag (die));
Packit 032894
Packit 032894
	  Dwarf_Addr lowpc, highpc;
Packit 032894
	  if (dwarf_lowpc (die, &lowpc) == 0
Packit 032894
	      && dwarf_highpc (die, &highpc) == 0)
Packit 032894
	    {
Packit 032894
	      lowpc += a->dwbias;
Packit 032894
	      highpc += a->dwbias;
Packit 032894
	      Dwfl_Line *loline = dwfl_getsrc (a->dwfl, lowpc);
Packit 032894
	      Dwfl_Line *hiline = dwfl_getsrc (a->dwfl, highpc - 1);
Packit 032894
	      paddr (": ", lowpc, loline);
Packit 032894
	      if (highpc != lowpc)
Packit 032894
		paddr (" .. ", highpc - 1, hiline == loline ? NULL : hiline);
Packit 032894
	    }
Packit 032894
	  puts ("");
Packit 032894
Packit 032894
	  print_vars (indent + INDENT, die);
Packit 032894
	}
Packit 032894
      free (scopes);
Packit 032894
    }
Packit 032894
Packit 032894
  return 0;
Packit 032894
}
Packit 032894
Packit 032894
Packit 032894
int
Packit 032894
main (int argc, char *argv[])
Packit 032894
{
Packit 032894
  int remaining;
Packit 032894
Packit 032894
  /* Set locale.  */
Packit 032894
  (void) setlocale (LC_ALL, "");
Packit 032894
Packit 032894
  struct args a = { .dwfl = NULL, .cu = NULL };
Packit 032894
Packit 032894
  (void) argp_parse (dwfl_standard_argp (), argc, argv, 0, &remaining,
Packit 032894
		     &a.dwfl);
Packit 032894
  assert (a.dwfl != NULL);
Packit 032894
  a.argv = &argv[remaining];
Packit 032894
Packit 032894
  int result = 0;
Packit 032894
Packit 032894
  while ((a.cu = dwfl_nextcu (a.dwfl, a.cu, &a.dwbias)) != NULL)
Packit 032894
    dwarf_getfuncs (a.cu, &handle_function, &a, 0);
Packit 032894
Packit 032894
  dwfl_end (a.dwfl);
Packit 032894
Packit 032894
  return result;
Packit 032894
}