Blame libdw/dwarf_getscopevar.c

Packit Service 97d2fb
/* Find a named variable or parameter within given scopes.
Packit Service 97d2fb
   Copyright (C) 2005-2009 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 either
Packit Service 97d2fb
Packit Service 97d2fb
     * the GNU Lesser General Public License as published by the Free
Packit Service 97d2fb
       Software Foundation; either version 3 of the License, or (at
Packit Service 97d2fb
       your option) any later version
Packit Service 97d2fb
Packit Service 97d2fb
   or
Packit Service 97d2fb
Packit Service 97d2fb
     * the GNU General Public License as published by the Free
Packit Service 97d2fb
       Software Foundation; either version 2 of the License, or (at
Packit Service 97d2fb
       your option) any later version
Packit Service 97d2fb
Packit Service 97d2fb
   or both in parallel, as here.
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 GNU
Packit Service 97d2fb
   General Public License for more details.
Packit Service 97d2fb
Packit Service 97d2fb
   You should have received copies of the GNU General Public License and
Packit Service 97d2fb
   the GNU Lesser General Public License along with this program.  If
Packit Service 97d2fb
   not, see <http://www.gnu.org/licenses/>.  */
Packit Service 97d2fb
Packit Service 97d2fb
#ifdef HAVE_CONFIG_H
Packit Service 97d2fb
# include <config.h>
Packit Service 97d2fb
#endif
Packit Service 97d2fb
Packit Service 97d2fb
#include <stdbool.h>
Packit Service 97d2fb
#include <string.h>
Packit Service 97d2fb
#include "libdwP.h"
Packit Service 97d2fb
#include <dwarf.h>
Packit Service 97d2fb
Packit Service 97d2fb
Packit Service 97d2fb
/* Find the containing CU's files.  */
Packit Service 97d2fb
static int
Packit Service 97d2fb
getfiles (Dwarf_Die *die, Dwarf_Files **files)
Packit Service 97d2fb
{
Packit Service 97d2fb
  return INTUSE(dwarf_getsrcfiles) (&CUDIE (die->cu), files, NULL);
Packit Service 97d2fb
}
Packit Service 97d2fb
Packit Service 97d2fb
/* Fetch an attribute that should have a constant integer form.  */
Packit Service 97d2fb
static int
Packit Service 97d2fb
getattr (Dwarf_Die *die, int search_name, Dwarf_Word *value)
Packit Service 97d2fb
{
Packit Service 97d2fb
  Dwarf_Attribute attr_mem;
Packit Service 97d2fb
  return INTUSE(dwarf_formudata) (INTUSE(dwarf_attr) (die, search_name,
Packit Service 97d2fb
						      &attr_mem), value);
Packit Service 97d2fb
}
Packit Service 97d2fb
Packit Service 97d2fb
static inline int
Packit Service 97d2fb
file_matches (const char *lastfile,
Packit Service 97d2fb
              size_t match_file_len, const char *match_file,
Packit Service 97d2fb
              Dwarf_Files *files, size_t idx,
Packit Service 97d2fb
              bool *lastfile_matches)
Packit Service 97d2fb
{
Packit Service 97d2fb
  if (idx >= files->nfiles)
Packit Service 97d2fb
    return false;
Packit Service 97d2fb
  const char *file = files->info[idx].name;
Packit Service 97d2fb
  if (file != lastfile)
Packit Service 97d2fb
    {
Packit Service 97d2fb
      size_t len = strlen (file);
Packit Service 97d2fb
      *lastfile_matches = (len >= match_file_len
Packit Service 97d2fb
                          && !memcmp (match_file, file, match_file_len)
Packit Service 97d2fb
                          && (len == match_file_len
Packit Service 97d2fb
                              || file[len - match_file_len - 1] == '/'));
Packit Service 97d2fb
    }
Packit Service 97d2fb
  return *lastfile_matches;
Packit Service 97d2fb
}
Packit Service 97d2fb
Packit Service 97d2fb
/* Search SCOPES[0..NSCOPES-1] for a variable called NAME.
Packit Service 97d2fb
   Ignore the first SKIP_SHADOWS scopes that match the name.
Packit Service 97d2fb
   If MATCH_FILE is not null, accept only declaration in that source file;
Packit Service 97d2fb
   if MATCH_LINENO or MATCH_LINECOL are also nonzero, accept only declaration
Packit Service 97d2fb
   at that line and column.
Packit Service 97d2fb
Packit Service 97d2fb
   If successful, fill in *RESULT with the DIE of the variable found,
Packit Service 97d2fb
   and return N where SCOPES[N] is the scope defining the variable.
Packit Service 97d2fb
   Return -1 for errors or -2 for no matching variable found.  */
Packit Service 97d2fb
Packit Service 97d2fb
int
Packit Service 97d2fb
dwarf_getscopevar (Dwarf_Die *scopes, int nscopes,
Packit Service 97d2fb
		   const char *name, int skip_shadows,
Packit Service 97d2fb
		   const char *match_file, int match_lineno, int match_linecol,
Packit Service 97d2fb
		   Dwarf_Die *result)
Packit Service 97d2fb
{
Packit Service 97d2fb
  /* Match against the given file name.  */
Packit Service 97d2fb
  size_t match_file_len = match_file == NULL ? 0 : strlen (match_file);
Packit Service 97d2fb
  bool lastfile_matches = false;
Packit Service 97d2fb
  const char *lastfile = NULL;
Packit Service 97d2fb
Packit Service 97d2fb
  /* Start with the innermost scope and move out.  */
Packit Service 97d2fb
  for (int out = 0; out < nscopes; ++out)
Packit Service 97d2fb
    if (INTUSE(dwarf_haschildren) (&scopes[out]))
Packit Service 97d2fb
      {
Packit Service 97d2fb
	if (INTUSE(dwarf_child) (&scopes[out], result) != 0)
Packit Service 97d2fb
	  return -1;
Packit Service 97d2fb
	do
Packit Service 97d2fb
	  {
Packit Service 97d2fb
	    switch (INTUSE(dwarf_tag) (result))
Packit Service 97d2fb
	      {
Packit Service 97d2fb
	      case DW_TAG_variable:
Packit Service 97d2fb
	      case DW_TAG_formal_parameter:
Packit Service 97d2fb
		break;
Packit Service 97d2fb
Packit Service 97d2fb
	      default:
Packit Service 97d2fb
		continue;
Packit Service 97d2fb
	      }
Packit Service 97d2fb
Packit Service 97d2fb
	    /* Only get here for a variable or parameter.  Check the name.  */
Packit Service 97d2fb
	    const char *diename = INTUSE(dwarf_diename) (result);
Packit Service 97d2fb
	    if (diename != NULL && !strcmp (name, diename))
Packit Service 97d2fb
	      {
Packit Service 97d2fb
		/* We have a matching name.  */
Packit Service 97d2fb
Packit Service 97d2fb
		if (skip_shadows > 0)
Packit Service 97d2fb
		  {
Packit Service 97d2fb
		    /* Punt this scope for the one it shadows.  */
Packit Service 97d2fb
		    --skip_shadows;
Packit Service 97d2fb
		    break;
Packit Service 97d2fb
		  }
Packit Service 97d2fb
Packit Service 97d2fb
		if (match_file != NULL)
Packit Service 97d2fb
		  {
Packit Service 97d2fb
		    /* Check its decl_file.  */
Packit Service 97d2fb
Packit Service 97d2fb
		    Dwarf_Word i;
Packit Service 97d2fb
		    Dwarf_Files *files;
Packit Service 97d2fb
		    if (getattr (result, DW_AT_decl_file, &i) != 0
Packit Service 97d2fb
			|| getfiles (&scopes[out], &files) != 0)
Packit Service 97d2fb
		      break;
Packit Service 97d2fb
Packit Service 97d2fb
		    if (!file_matches (lastfile, match_file_len, match_file,
Packit Service 97d2fb
		                       files, i, &lastfile_matches))
Packit Service 97d2fb
		      break;
Packit Service 97d2fb
Packit Service 97d2fb
		    if (match_lineno > 0
Packit Service 97d2fb
			&& (getattr (result, DW_AT_decl_line, &i) != 0
Packit Service 97d2fb
			    || (int) i != match_lineno))
Packit Service 97d2fb
		      break;
Packit Service 97d2fb
		    if (match_linecol > 0
Packit Service 97d2fb
			&& (getattr (result, DW_AT_decl_column, &i) != 0
Packit Service 97d2fb
			    || (int) i != match_linecol))
Packit Service 97d2fb
		      break;
Packit Service 97d2fb
		  }
Packit Service 97d2fb
Packit Service 97d2fb
		/* We have a winner!  */
Packit Service 97d2fb
		return out;
Packit Service 97d2fb
	      }
Packit Service 97d2fb
	  }
Packit Service 97d2fb
	while (INTUSE(dwarf_siblingof) (result, result) == 0);
Packit Service 97d2fb
      }
Packit Service 97d2fb
Packit Service 97d2fb
  return -2;
Packit Service 97d2fb
}