Blame debug/backtracesyms.c

Packit Service 82fcde
/* Return list with names for address in backtrace.
Packit Service 82fcde
   Copyright (C) 1998-2018 Free Software Foundation, Inc.
Packit Service 82fcde
   This file is part of the GNU C Library.
Packit Service 82fcde
   Contributed by Ulrich Drepper <drepper@cygnus.com>, 1998.
Packit Service 82fcde
Packit Service 82fcde
   The GNU C Library is free software; you can redistribute it and/or
Packit Service 82fcde
   modify it under the terms of the GNU Lesser General Public
Packit Service 82fcde
   License as published by the Free Software Foundation; either
Packit Service 82fcde
   version 2.1 of the License, or (at your option) any later version.
Packit Service 82fcde
Packit Service 82fcde
   The GNU C Library is distributed in the hope that it will be useful,
Packit Service 82fcde
   but WITHOUT ANY WARRANTY; without even the implied warranty of
Packit Service 82fcde
   MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
Packit Service 82fcde
   Lesser General Public License for more details.
Packit Service 82fcde
Packit Service 82fcde
   You should have received a copy of the GNU Lesser General Public
Packit Service 82fcde
   License along with the GNU C Library; if not, see
Packit Service 82fcde
   <http://www.gnu.org/licenses/>.  */
Packit Service 82fcde
Packit Service 82fcde
#include <assert.h>
Packit Service 82fcde
#include <execinfo.h>
Packit Service 82fcde
#include <stdio.h>
Packit Service 82fcde
#include <stdlib.h>
Packit Service 82fcde
#include <string.h>
Packit Service 82fcde
Packit Service 82fcde
#include <ldsodefs.h>
Packit Service 82fcde
Packit Service 82fcde
#if __ELF_NATIVE_CLASS == 32
Packit Service 82fcde
# define WORD_WIDTH 8
Packit Service 82fcde
#else
Packit Service 82fcde
/* We assume 64bits.  */
Packit Service 82fcde
# define WORD_WIDTH 16
Packit Service 82fcde
#endif
Packit Service 82fcde
Packit Service 82fcde
Packit Service 82fcde
char **
Packit Service 82fcde
__backtrace_symbols (void *const *array, int size)
Packit Service 82fcde
{
Packit Service 82fcde
  Dl_info info[size];
Packit Service 82fcde
  int status[size];
Packit Service 82fcde
  int cnt;
Packit Service 82fcde
  size_t total = 0;
Packit Service 82fcde
  char **result;
Packit Service 82fcde
Packit Service 82fcde
  /* Fill in the information we can get from `dladdr'.  */
Packit Service 82fcde
  for (cnt = 0; cnt < size; ++cnt)
Packit Service 82fcde
    {
Packit Service 82fcde
      struct link_map *map;
Packit Service 82fcde
      status[cnt] = _dl_addr (array[cnt], &info[cnt], &map, NULL);
Packit Service 82fcde
      if (status[cnt] && info[cnt].dli_fname && info[cnt].dli_fname[0] != '\0')
Packit Service 82fcde
	{
Packit Service 82fcde
	  /* We have some info, compute the length of the string which will be
Packit Service 82fcde
	     "<file-name>(<sym-name>+offset) [address].  */
Packit Service 82fcde
	  total += (strlen (info[cnt].dli_fname ?: "")
Packit Service 82fcde
		    + strlen (info[cnt].dli_sname ?: "")
Packit Service 82fcde
		    + 3 + WORD_WIDTH + 3 + WORD_WIDTH + 5);
Packit Service 82fcde
Packit Service 82fcde
	  /* The load bias is more useful to the user than the load
Packit Service 82fcde
	     address.  The use of these addresses is to calculate an
Packit Service 82fcde
	     address in the ELF file, so its prelinked bias is not
Packit Service 82fcde
	     something we want to subtract out.  */
Packit Service 82fcde
	  info[cnt].dli_fbase = (void *) map->l_addr;
Packit Service 82fcde
	}
Packit Service 82fcde
      else
Packit Service 82fcde
	total += 5 + WORD_WIDTH;
Packit Service 82fcde
    }
Packit Service 82fcde
Packit Service 82fcde
  /* Allocate memory for the result.  */
Packit Service 82fcde
  result = (char **) malloc (size * sizeof (char *) + total);
Packit Service 82fcde
  if (result != NULL)
Packit Service 82fcde
    {
Packit Service 82fcde
      char *last = (char *) (result + size);
Packit Service 82fcde
Packit Service 82fcde
      for (cnt = 0; cnt < size; ++cnt)
Packit Service 82fcde
	{
Packit Service 82fcde
	  result[cnt] = last;
Packit Service 82fcde
Packit Service 82fcde
	  if (status[cnt]
Packit Service 82fcde
	      && info[cnt].dli_fname != NULL && info[cnt].dli_fname[0] != '\0')
Packit Service 82fcde
	    {
Packit Service 82fcde
	      if (info[cnt].dli_sname == NULL)
Packit Service 82fcde
		/* We found no symbol name to use, so describe it as
Packit Service 82fcde
		   relative to the file.  */
Packit Service 82fcde
		info[cnt].dli_saddr = info[cnt].dli_fbase;
Packit Service 82fcde
Packit Service 82fcde
	      if (info[cnt].dli_sname == NULL && info[cnt].dli_saddr == 0)
Packit Service 82fcde
		last += 1 + sprintf (last, "%s(%s) [%p]",
Packit Service 82fcde
				     info[cnt].dli_fname ?: "",
Packit Service 82fcde
				     info[cnt].dli_sname ?: "",
Packit Service 82fcde
				     array[cnt]);
Packit Service 82fcde
	      else
Packit Service 82fcde
		{
Packit Service 82fcde
		  char sign;
Packit Service 82fcde
		  ptrdiff_t offset;
Packit Service 82fcde
		  if (array[cnt] >= (void *) info[cnt].dli_saddr)
Packit Service 82fcde
		    {
Packit Service 82fcde
		      sign = '+';
Packit Service 82fcde
		      offset = array[cnt] - info[cnt].dli_saddr;
Packit Service 82fcde
		    }
Packit Service 82fcde
		  else
Packit Service 82fcde
		    {
Packit Service 82fcde
		      sign = '-';
Packit Service 82fcde
		      offset = info[cnt].dli_saddr - array[cnt];
Packit Service 82fcde
		    }
Packit Service 82fcde
Packit Service 82fcde
		  last += 1 + sprintf (last, "%s(%s%c%#tx) [%p]",
Packit Service 82fcde
				       info[cnt].dli_fname ?: "",
Packit Service 82fcde
				       info[cnt].dli_sname ?: "",
Packit Service 82fcde
				       sign, offset, array[cnt]);
Packit Service 82fcde
		}
Packit Service 82fcde
	    }
Packit Service 82fcde
	  else
Packit Service 82fcde
	    last += 1 + sprintf (last, "[%p]", array[cnt]);
Packit Service 82fcde
	}
Packit Service 82fcde
      assert (last <= (char *) result + size * sizeof (char *) + total);
Packit Service 82fcde
    }
Packit Service 82fcde
Packit Service 82fcde
  return result;
Packit Service 82fcde
}
Packit Service 82fcde
weak_alias (__backtrace_symbols, backtrace_symbols)