Blame libdwfl/dwfl_module_getsrc_file.c

Packit Service 97d2fb
/* Find matching source locations in a module.
Packit Service 97d2fb
   Copyright (C) 2005 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 "libdwflP.h"
Packit Service 97d2fb
#include "../libdw/libdwP.h"
Packit Service 97d2fb
Packit Service 97d2fb
Packit Service 97d2fb
static inline const char *
Packit Service 97d2fb
dwfl_dwarf_line_file (const Dwarf_Line *line)
Packit Service 97d2fb
{
Packit Service 97d2fb
  return line->files->info[line->file].name;
Packit Service 97d2fb
}
Packit Service 97d2fb
Packit Service 97d2fb
static inline Dwarf_Line *
Packit Service 97d2fb
dwfl_line (const Dwfl_Line *line)
Packit Service 97d2fb
{
Packit Service 97d2fb
  return &dwfl_linecu (line)->die.cu->lines->info[line->idx];
Packit Service 97d2fb
}
Packit Service 97d2fb
Packit Service 97d2fb
static inline const char *
Packit Service 97d2fb
dwfl_line_file (const Dwfl_Line *line)
Packit Service 97d2fb
{
Packit Service 97d2fb
  return dwfl_dwarf_line_file (dwfl_line (line));
Packit Service 97d2fb
}
Packit Service 97d2fb
Packit Service 97d2fb
int
Packit Service 97d2fb
dwfl_module_getsrc_file (Dwfl_Module *mod,
Packit Service 97d2fb
			 const char *fname, int lineno, int column,
Packit Service 97d2fb
			 Dwfl_Line ***srcsp, size_t *nsrcs)
Packit Service 97d2fb
{
Packit Service 97d2fb
  if (mod == NULL)
Packit Service 97d2fb
    return -1;
Packit Service 97d2fb
Packit Service 97d2fb
  if (mod->dw == NULL)
Packit Service 97d2fb
    {
Packit Service 97d2fb
      Dwarf_Addr bias;
Packit Service 97d2fb
      if (INTUSE(dwfl_module_getdwarf) (mod, &bias) == NULL)
Packit Service 97d2fb
	return -1;
Packit Service 97d2fb
    }
Packit Service 97d2fb
Packit Service 97d2fb
  bool is_basename = strchr (fname, '/') == NULL;
Packit Service 97d2fb
Packit Service 97d2fb
  size_t max_match = *nsrcs ?: ~0u;
Packit Service 97d2fb
  size_t act_match = *nsrcs;
Packit Service 97d2fb
  size_t cur_match = 0;
Packit Service 97d2fb
  Dwfl_Line **match = *nsrcs == 0 ? NULL : *srcsp;
Packit Service 97d2fb
Packit Service 97d2fb
  struct dwfl_cu *cu = NULL;
Packit Service 97d2fb
  Dwfl_Error error;
Packit Service 97d2fb
  while ((error = __libdwfl_nextcu (mod, cu, &cu)) == DWFL_E_NOERROR
Packit Service 97d2fb
	 && cu != NULL
Packit Service 97d2fb
	 && (error = __libdwfl_cu_getsrclines (cu)) == DWFL_E_NOERROR)
Packit Service 97d2fb
    {
Packit Service 97d2fb
      /* Search through all the line number records for a matching
Packit Service 97d2fb
	 file and line/column number.  If any of the numbers is zero,
Packit Service 97d2fb
	 no match is performed.  */
Packit Service 97d2fb
      const char *lastfile = NULL;
Packit Service 97d2fb
      bool lastmatch = false;
Packit Service 97d2fb
      for (size_t cnt = 0; cnt < cu->die.cu->lines->nlines; ++cnt)
Packit Service 97d2fb
	{
Packit Service 97d2fb
	  Dwarf_Line *line = &cu->die.cu->lines->info[cnt];
Packit Service 97d2fb
Packit Service 97d2fb
	  if (unlikely (line->file >= line->files->nfiles))
Packit Service 97d2fb
	    {
Packit Service 97d2fb
	      if (*nsrcs == 0)
Packit Service 97d2fb
		free (match);
Packit Service 97d2fb
	      __libdwfl_seterrno (DWFL_E (LIBDW, DWARF_E_INVALID_DWARF));
Packit Service 97d2fb
	      return -1;
Packit Service 97d2fb
	    }
Packit Service 97d2fb
	  else
Packit Service 97d2fb
	    {
Packit Service 97d2fb
	      const char *file = dwfl_dwarf_line_file (line);
Packit Service 97d2fb
	      if (file != lastfile)
Packit Service 97d2fb
		{
Packit Service 97d2fb
		  /* Match the name with the name the user provided.  */
Packit Service 97d2fb
		  lastfile = file;
Packit Service 97d2fb
		  lastmatch = !strcmp (is_basename ? basename (file) : file,
Packit Service 97d2fb
				       fname);
Packit Service 97d2fb
		}
Packit Service 97d2fb
	    }
Packit Service 97d2fb
	  if (!lastmatch)
Packit Service 97d2fb
	    continue;
Packit Service 97d2fb
Packit Service 97d2fb
	  /* See whether line and possibly column match.  */
Packit Service 97d2fb
	  if (lineno != 0
Packit Service 97d2fb
	      && (lineno > line->line
Packit Service 97d2fb
		  || (column != 0 && column > line->column)))
Packit Service 97d2fb
	    /* Cannot match.  */
Packit Service 97d2fb
	    continue;
Packit Service 97d2fb
Packit Service 97d2fb
	  /* Determine whether this is the best match so far.  */
Packit Service 97d2fb
	  size_t inner;
Packit Service 97d2fb
	  for (inner = 0; inner < cur_match; ++inner)
Packit Service 97d2fb
	    if (dwfl_line_file (match[inner])
Packit Service 97d2fb
		== dwfl_dwarf_line_file (line))
Packit Service 97d2fb
	      break;
Packit Service 97d2fb
	  if (inner < cur_match
Packit Service 97d2fb
	      && (dwfl_line (match[inner])->line != line->line
Packit Service 97d2fb
		  || dwfl_line (match[inner])->line != lineno
Packit Service 97d2fb
		  || (column != 0
Packit Service 97d2fb
		      && (dwfl_line (match[inner])->column != line->column
Packit Service 97d2fb
			  || dwfl_line (match[inner])->column != column))))
Packit Service 97d2fb
	    {
Packit Service 97d2fb
	      /* We know about this file already.  If this is a better
Packit Service 97d2fb
		 match for the line number, use it.  */
Packit Service 97d2fb
	      if (dwfl_line (match[inner])->line >= line->line
Packit Service 97d2fb
		  && (dwfl_line (match[inner])->line != line->line
Packit Service 97d2fb
		      || dwfl_line (match[inner])->column >= line->column))
Packit Service 97d2fb
		/* Use the new line.  Otherwise the old one.  */
Packit Service 97d2fb
		match[inner] = &cu->lines->idx[cnt];
Packit Service 97d2fb
	      continue;
Packit Service 97d2fb
	    }
Packit Service 97d2fb
Packit Service 97d2fb
	  if (cur_match < max_match)
Packit Service 97d2fb
	    {
Packit Service 97d2fb
	      if (cur_match == act_match)
Packit Service 97d2fb
		{
Packit Service 97d2fb
		  /* Enlarge the array for the results.  */
Packit Service 97d2fb
		  act_match += 10;
Packit Service 97d2fb
		  Dwfl_Line **newp = realloc (match,
Packit Service 97d2fb
					      act_match
Packit Service 97d2fb
					      * sizeof (Dwfl_Line *));
Packit Service 97d2fb
		  if (newp == NULL)
Packit Service 97d2fb
		    {
Packit Service 97d2fb
		      free (match);
Packit Service 97d2fb
		      __libdwfl_seterrno (DWFL_E_NOMEM);
Packit Service 97d2fb
		      return -1;
Packit Service 97d2fb
		    }
Packit Service 97d2fb
		  match = newp;
Packit Service 97d2fb
		}
Packit Service 97d2fb
Packit Service 97d2fb
	      match[cur_match++] = &cu->lines->idx[cnt];
Packit Service 97d2fb
	    }
Packit Service 97d2fb
	}
Packit Service 97d2fb
    }
Packit Service 97d2fb
Packit Service 97d2fb
  if (cur_match > 0)
Packit Service 97d2fb
    {
Packit Service 97d2fb
      assert (*nsrcs == 0 || *srcsp == match);
Packit Service 97d2fb
Packit Service 97d2fb
      *nsrcs = cur_match;
Packit Service 97d2fb
      *srcsp = match;
Packit Service 97d2fb
Packit Service 97d2fb
      return 0;
Packit Service 97d2fb
    }
Packit Service 97d2fb
Packit Service 97d2fb
  __libdwfl_seterrno (DWFL_E_NO_MATCH);
Packit Service 97d2fb
  return -1;
Packit Service 97d2fb
}