/*
 pstack.c -- asynchronous stack trace of a running process
 Copyright (c) 1999 Ross Thompson
 Author: Ross Thompson <ross@whatsis.com>
 Critical bug fix: Tim Waugh
*/

/*
 This file is free software; you can redistribute it and/or modify
 it under the terms of the GNU General Public License as published by
 the Free Software Foundation; either version 2 of the License, or
 (at your option) any later version.

 This program is distributed in the hope that it will be useful,
 but WITHOUT ANY WARRANTY; without even the implied warranty of
 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
 GNU General Public License for more details.

 You should have received a copy of the GNU General Public License
 along with this program; if not, write to the Free Software
 Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
*/

/* RESTRICTIONS:

   pstack currently works only on Linux, only on an x86 machine running
   32 bit ELF binaries (64 bit not supported).  Also, for symbolic
   information, you need to use a GNU compiler to generate your
   program, and you can't strip symbols from the binaries.  For thread
   information to be dumped, you have to use the debug-aware version
   of libpthread.so.  (To check, run 'nm' on your libpthread.so, and
   make sure that the symbol "__pthread_threads_debug" is defined.)

   The details of pulling stuff out of ELF files and running through
   program images is very platform specific, and I don't want to
   try to support modes or machine types I can't test in or on.
   If someone wants to generalize this to other architectures, I would
   be happy to help and coordinate the activity.  Please send me whatever
   changes you make to support these machines, so that I can own the
   central font of all truth (at least as regards this program).

   Thanks 
*/

#include <sys/types.h>
#include <sys/stat.h>
#include <sys/wait.h>
#include <sys/ptrace.h>
#include <asm/ptrace.h>

#include <assert.h>
#include <fcntl.h>
#include <link.h>
#include <malloc.h>
#include <string.h>
#include <stdarg.h>
#include <unistd.h>
#include <stdlib.h>
#include <stdio.h>
#include <errno.h>
#include <signal.h>
#include <pthread.h>
#include <limits.h>		/* PTHREAD_THREADS_MAX */


#include <bfd.h>

#include "libiberty.h"

#include "pstack.h"		/* just one function */
#include "budbg.h"		/* binutils stuff related to debugging symbols. */
#include "bucomm.h"		/* some common stuff */
#include "debug.h"		/* and more binutils stuff... */
#include "budbg.h"
#include "linuxthreads.h"	/* LinuxThreads specific stuff... */


/*
 * fprintf for file descriptors :) NOTE: we have to use fixed-size buffer :)(
 * due to malloc's unavalaibility.
 */
int
fdprintf(	int		fd,
		const char*	fmt,...)
{
	char	xbuf[2048];// FIXME: enough?
	va_list	ap;
	int	r;
	if (fd<0)
		return -1;
	va_start(ap, fmt);
	r = vsnprintf(xbuf, sizeof(xbuf), fmt, ap);
	va_end(ap);
	return write(fd, xbuf, r);
}

int
fdputc(		char		c,
		int		fd)
{
	if (fd<0)
		return -1;
	return write(fd, &c, sizeof(c));
}

int
fdputs(		const char*	s,
		int		fd)
{
	if (fd<0)
		return -1;
	return write(fd, s, strlen(s));
}

/*
 * Use this function to open log file.
 * Flags: truncate on opening.
 */
static	const char*		path_format = "stack-trace-on-segv-%d.txt";
static int
open_log_file(	const pthread_t	tid,
		const pid_t	pid)
{
	char	fname[PATH_MAX];
	int	r;
	snprintf(fname, sizeof(fname), path_format, tid, pid);
	r = open(fname, O_WRONLY|O_CREAT|O_TRUNC,
			S_IRUSR|S_IWUSR);
	if (r<0)
		perror("open");
	return r;
}
/*
 * Add additional debugging information for functions.
 */

/*
 * Lineno
 */
typedef struct {
	int			lineno;
	bfd_vma			addr;
} debug_lineno_t;

/*
 * Block - a {} pair.
 */
typedef struct debug_block_st {
	bfd_vma			begin_addr;	/* where did it start */
	bfd_vma			end_addr;	/* where did it end */
	struct debug_block_st*	parent;
	struct debug_block_st*	childs;
	int			childs_count;
} debug_block_t;

/*
 * Function parameter.
 */
typedef struct {
	bfd_vma			offset;	/* Offset in the stack */
	const char*		name;	/* And name. */
} debug_parameter_t;

/*
 * Extra information about functions.
 */
typedef struct {
	asymbol*		symbol;	/* mangled function name, addr */
	debug_lineno_t*		lines;
	int			lines_count;
	int			max_lines_count;
	const char*		name;
	const char*		filename;/* a file name it occured in... */
	debug_block_t*		block;	/* each function has a block, or not, you know */
	debug_parameter_t*	argv;	/* argument types. */
	int			argc;
	int			max_argc;
} debug_function_t;

/* This is the structure we use as a handle for these routines.  */
struct pr_handle
{
  /* File to print information to.  */
  FILE *f;
  /* Current indentation level.  */
  unsigned int indent;
  /* Type stack.  */
  struct pr_stack *stack;
  /* Parameter number we are about to output.  */
  int parameter;
  debug_block_t*	block;			/* current block */
  debug_function_t*	function;		/* current function */
  debug_function_t*	functions;		/* all functions */
  int			functions_size;		/* current size */
  int			functions_maxsize;	/* maximum size */
};

/* The type stack.  */

struct pr_stack
{
  /* Next element on the stack.  */
  struct pr_stack *next;
  /* This element.  */
  char *type;
  /* Current visibility of fields if this is a class.  */
  enum debug_visibility visibility;
  /* Name of the current method we are handling.  */
  const char *method;
};

static void indent PARAMS ((struct pr_handle *));
static boolean push_type PARAMS ((struct pr_handle *, const char *));
static boolean prepend_type PARAMS ((struct pr_handle *, const char *));
static boolean append_type PARAMS ((struct pr_handle *, const char *));
static boolean substitute_type PARAMS ((struct pr_handle *, const char *));
static boolean indent_type PARAMS ((struct pr_handle *));
static char *pop_type PARAMS ((struct pr_handle *));
static void print_vma PARAMS ((bfd_vma, char *, boolean, boolean));
static boolean pr_fix_visibility
  PARAMS ((struct pr_handle *, enum debug_visibility));

static boolean pr_start_compilation_unit PARAMS ((PTR, const char *));
static boolean pr_start_source PARAMS ((PTR, const char *));
static boolean pr_empty_type PARAMS ((PTR));
static boolean pr_void_type PARAMS ((PTR));
static boolean pr_int_type PARAMS ((PTR, unsigned int, boolean));
static boolean pr_float_type PARAMS ((PTR, unsigned int));
static boolean pr_complex_type PARAMS ((PTR, unsigned int));
static boolean pr_bool_type PARAMS ((PTR, unsigned int));
static boolean pr_enum_type
  PARAMS ((PTR, const char *, const char **, bfd_signed_vma *));
static boolean pr_pointer_type PARAMS ((PTR));
static boolean pr_function_type PARAMS ((PTR, int, boolean));
static boolean pr_reference_type PARAMS ((PTR));
static boolean pr_range_type PARAMS ((PTR, bfd_signed_vma, bfd_signed_vma));
static boolean pr_array_type
  PARAMS ((PTR, bfd_signed_vma, bfd_signed_vma, boolean));
static boolean pr_set_type PARAMS ((PTR, boolean));
static boolean pr_offset_type PARAMS ((PTR));
static boolean pr_method_type PARAMS ((PTR, boolean, int, boolean));
static boolean pr_const_type PARAMS ((PTR));
static boolean pr_volatile_type PARAMS ((PTR));
static boolean pr_start_struct_type
  PARAMS ((PTR, const char *, unsigned int, boolean, unsigned int));
static boolean pr_struct_field
  PARAMS ((PTR, const char *, bfd_vma, bfd_vma, enum debug_visibility));
static boolean pr_end_struct_type PARAMS ((PTR));
static boolean pr_start_class_type
  PARAMS ((PTR, const char *, unsigned int, boolean, unsigned int, boolean,
	   boolean));
static boolean pr_class_static_member
  PARAMS ((PTR, const char *, const char *, enum debug_visibility));
static boolean pr_class_baseclass
  PARAMS ((PTR, bfd_vma, boolean, enum debug_visibility));
static boolean pr_class_start_method PARAMS ((PTR, const char *));
static boolean pr_class_method_variant
  PARAMS ((PTR, const char *, enum debug_visibility, boolean, boolean,
	   bfd_vma, boolean));
static boolean pr_class_static_method_variant
  PARAMS ((PTR, const char *, enum debug_visibility, boolean, boolean));
static boolean pr_class_end_method PARAMS ((PTR));
static boolean pr_end_class_type PARAMS ((PTR));
static boolean pr_typedef_type PARAMS ((PTR, const char *));
static boolean pr_tag_type
  PARAMS ((PTR, const char *, unsigned int, enum debug_type_kind));
static boolean pr_typdef PARAMS ((PTR, const char *));
static boolean pr_tag PARAMS ((PTR, const char *));
static boolean pr_int_constant PARAMS ((PTR, const char *, bfd_vma));
static boolean pr_float_constant PARAMS ((PTR, const char *, double));
static boolean pr_typed_constant PARAMS ((PTR, const char *, bfd_vma));
static boolean pr_variable
  PARAMS ((PTR, const char *, enum debug_var_kind, bfd_vma));
static boolean pr_start_function PARAMS ((PTR, const char *, boolean));
static boolean pr_function_parameter
  PARAMS ((PTR, const char *, enum debug_parm_kind, bfd_vma));
static boolean pr_start_block PARAMS ((PTR, bfd_vma));
static boolean pr_end_block PARAMS ((PTR, bfd_vma));
static boolean pr_end_function PARAMS ((PTR));
static boolean pr_lineno PARAMS ((PTR, const char *, unsigned long, bfd_vma));

static const struct debug_write_fns pr_fns =
{
  pr_start_compilation_unit,
  pr_start_source,
  pr_empty_type,
  pr_void_type,
  pr_int_type,
  pr_float_type,
  pr_complex_type,
  pr_bool_type,
  pr_enum_type,
  pr_pointer_type,
  pr_function_type,
  pr_reference_type,
  pr_range_type,
  pr_array_type,
  pr_set_type,
  pr_offset_type,
  pr_method_type,
  pr_const_type,
  pr_volatile_type,
  pr_start_struct_type,
  pr_struct_field,
  pr_end_struct_type,
  pr_start_class_type,
  pr_class_static_member,
  pr_class_baseclass,
  pr_class_start_method,
  pr_class_method_variant,
  pr_class_static_method_variant,
  pr_class_end_method,
  pr_end_class_type,
  pr_typedef_type,
  pr_tag_type,
  pr_typdef,
  pr_tag,
  pr_int_constant,
  pr_float_constant,
  pr_typed_constant,
  pr_variable,
  pr_start_function,
  pr_function_parameter,
  pr_start_block,
  pr_end_block,
  pr_end_function,
  pr_lineno
};


/* Indent to the current indentation level.  */

static void
indent (info)
     struct pr_handle *info;
{
  unsigned int i;

  for (i = 0; i < info->indent; i++)
    TRACE_PUTC ((' ', info->f));
}

/* Push a type on the type stack.  */

static boolean
push_type (info, type)
     struct pr_handle *info;
     const char *type;
{
  struct pr_stack *n;

  if (type == NULL)
    return false;

  n = (struct pr_stack *) xmalloc (sizeof *n);
  memset (n, 0, sizeof *n);

  n->type = xstrdup (type);
  n->visibility = DEBUG_VISIBILITY_IGNORE;
  n->method = NULL;
  n->next = info->stack;
  info->stack = n;

  return true;
}

/* Prepend a string onto the type on the top of the type stack.  */

static boolean
prepend_type (info, s)
     struct pr_handle *info;
     const char *s;
{
  char *n;

  assert (info->stack != NULL);

  n = (char *) xmalloc (strlen (s) + strlen (info->stack->type) + 1);
  sprintf (n, "%s%s", s, info->stack->type);
  free (info->stack->type);
  info->stack->type = n;

  return true;
}

/* Append a string to the type on the top of the type stack.  */

static boolean
append_type (info, s)
     struct pr_handle *info;
     const char *s;
{
  unsigned int len;

  if (s == NULL)
    return false;

  assert (info->stack != NULL);

  len = strlen (info->stack->type);
  info->stack->type = (char *) xrealloc (info->stack->type,
					 len + strlen (s) + 1);
  strcpy (info->stack->type + len, s);

  return true;
}

/* We use an underscore to indicate where the name should go in a type
   string.  This function substitutes a string for the underscore.  If
   there is no underscore, the name follows the type.  */

static boolean
substitute_type (info, s)
     struct pr_handle *info;
     const char *s;
{
  char *u;

  assert (info->stack != NULL);

  u = strchr (info->stack->type, '|');
  if (u != NULL)
    {
      char *n;

      n = (char *) xmalloc (strlen (info->stack->type) + strlen (s));

      memcpy (n, info->stack->type, u - info->stack->type);
      strcpy (n + (u - info->stack->type), s);
      strcat (n, u + 1);

      free (info->stack->type);
      info->stack->type = n;

      return true;
    }

  if (strchr (s, '|') != NULL
      && (strchr (info->stack->type, '{') != NULL
	  || strchr (info->stack->type, '(') != NULL))
    {
      if (! prepend_type (info, "(")
	  || ! append_type (info, ")"))
	return false;
    }

  if (*s == '\0')
    return true;

  return (append_type (info, " ")
	  && append_type (info, s));
}

/* Indent the type at the top of the stack by appending spaces.  */

static boolean
indent_type (info)
     struct pr_handle *info;
{
  unsigned int i;

  for (i = 0; i < info->indent; i++)
    {
      if (! append_type (info, " "))
	return false;
    }

  return true;
}

/* Pop a type from the type stack.  */

static char *
pop_type (info)
     struct pr_handle *info;
{
  struct pr_stack *o;
  char *ret;

  assert (info->stack != NULL);

  o = info->stack;
  info->stack = o->next;
  ret = o->type;
  free (o);

  return ret;
}

/* Print a VMA value into a string.  */

static void
print_vma (vma, buf, unsignedp, hexp)
     bfd_vma vma;
     char *buf;
     boolean unsignedp;
     boolean hexp;
{
  if (sizeof (vma) <= sizeof (unsigned long))
    {
      if (hexp)
	sprintf (buf, "0x%lx", (unsigned long) vma);
      else if (unsignedp)
	sprintf (buf, "%lu", (unsigned long) vma);
      else
	sprintf (buf, "%ld", (long) vma);
    }
  else
    {
      buf[0] = '0';
      buf[1] = 'x';
      sprintf_vma (buf + 2, vma);
    }
}

/* Start a new compilation unit.  */

static boolean
pr_start_compilation_unit (p, filename)
     PTR p;
     const char *filename;
{
  struct pr_handle *info = (struct pr_handle *) p;

  assert (info->indent == 0);
/*
  TRACE_FPRINTF( (info->f, "%s:\n", filename));
*/
  return true;
}

/* Start a source file within a compilation unit.  */

static boolean
pr_start_source (p, filename)
     PTR p;
     const char *filename;
{
  struct pr_handle *info = (struct pr_handle *) p;

  assert (info->indent == 0);
/*
  TRACE_FPRINTF( (info->f, " %s:\n", filename));
*/
  return true;
}

/* Push an empty type onto the type stack.  */

static boolean
pr_empty_type (p)
     PTR p;
{
  struct pr_handle *info = (struct pr_handle *) p;

  return push_type (info, "<undefined>");
}

/* Push a void type onto the type stack.  */

static boolean
pr_void_type (p)
     PTR p;
{
  struct pr_handle *info = (struct pr_handle *) p;

  return push_type (info, "void");
}

/* Push an integer type onto the type stack.  */

static boolean
pr_int_type (p, size, unsignedp)
     PTR p;
     unsigned int size;
     boolean unsignedp;
{
  struct pr_handle *info = (struct pr_handle *) p;
  char ab[10];

  sprintf (ab, "%sint%d", unsignedp ? "u" : "", size * 8);
  return push_type (info, ab);
}

/* Push a floating type onto the type stack.  */

static boolean
pr_float_type (p, size)
     PTR p;
     unsigned int size;
{
  struct pr_handle *info = (struct pr_handle *) p;
  char ab[10];

  if (size == 4)
    return push_type (info, "float");
  else if (size == 8)
    return push_type (info, "double");

  sprintf (ab, "float%d", size * 8);
  return push_type (info, ab);
}

/* Push a complex type onto the type stack.  */

static boolean
pr_complex_type (p, size)
     PTR p;
     unsigned int size;
{
  struct pr_handle *info = (struct pr_handle *) p;

  if (! pr_float_type (p, size))
    return false;

  return prepend_type (info, "complex ");
}

/* Push a boolean type onto the type stack.  */

static boolean
pr_bool_type (p, size)
     PTR p;
     unsigned int size;
{
  struct pr_handle *info = (struct pr_handle *) p;
  char ab[10];

  sprintf (ab, "bool%d", size * 8);

  return push_type (info, ab);
}

/* Push an enum type onto the type stack.  */

static boolean
pr_enum_type (p, tag, names, values)
     PTR p;
     const char *tag;
     const char **names;
     bfd_signed_vma *values;
{
  struct pr_handle *info = (struct pr_handle *) p;
  unsigned int i;
  bfd_signed_vma val;

  if (! push_type (info, "enum "))
    return false;
  if (tag != NULL)
    {
      if (! append_type (info, tag)
	  || ! append_type (info, " "))
	return false;
    }
  if (! append_type (info, "{ "))
    return false;

  if (names == NULL)
    {
      if (! append_type (info, "/* undefined */"))
	return false;
    }
  else
    {
      val = 0;
      for (i = 0; names[i] != NULL; i++)
	{
	  if (i > 0)
	    {
	      if (! append_type (info, ", "))
		return false;
	    }

	  if (! append_type (info, names[i]))
	    return false;

	  if (values[i] != val)
	    {
	      char ab[20];

	      print_vma (values[i], ab, false, false);
	      if (! append_type (info, " = ")
		  || ! append_type (info, ab))
		return false;
	      val = values[i];
	    }

	  ++val;
	}
    }

  return append_type (info, " }");
}

/* Turn the top type on the stack into a pointer.  */

static boolean
pr_pointer_type (p)
     PTR p;
{
  struct pr_handle *info = (struct pr_handle *) p;
  char *s;

  assert (info->stack != NULL);

  s = strchr (info->stack->type, '|');
  if (s != NULL && s[1] == '[')
    return substitute_type (info, "(*|)");
  return substitute_type (info, "*|");
}

/* Turn the top type on the stack into a function returning that type.  */

static boolean
pr_function_type (p, argcount, varargs)
     PTR p;
     int argcount;
     boolean varargs;
{
  struct pr_handle *info = (struct pr_handle *) p;
  char **arg_types;
  unsigned int len;
  char *s;

  assert (info->stack != NULL);

  len = 10;

  if (argcount <= 0)
    {
      arg_types = NULL;
      len += 15;
    }
  else
    {
      int i;

      arg_types = (char **) xmalloc (argcount * sizeof *arg_types);
      for (i = argcount - 1; i >= 0; i--)
	{
	  if (! substitute_type (info, ""))
	    return false;
	  arg_types[i] = pop_type (info);
	  if (arg_types[i] == NULL)
	    return false;
	  len += strlen (arg_types[i]) + 2;
	}
      if (varargs)
	len += 5;
    }

  /* Now the return type is on the top of the stack.  */

  s = (char *) xmalloc (len);
  strcpy (s, "(|) (");

  if (argcount < 0)
    {
#if 0
      /* Turn off unknown arguments. */
      strcat (s, "/* unknown */");
#endif
    }
  else
    {
      int i;

      for (i = 0; i < argcount; i++)
	{
	  if (i > 0)
	    strcat (s, ", ");
	  strcat (s, arg_types[i]);
	}
      if (varargs)
	{
	  if (i > 0)
	    strcat (s, ", ");
	  strcat (s, "...");
	}
      if (argcount > 0)
	free (arg_types);
    }

  strcat (s, ")");

  if (! substitute_type (info, s))
    return false;

  free (s);

  return true;
}

/* Turn the top type on the stack into a reference to that type.  */

static boolean
pr_reference_type (p)
     PTR p;
{
  struct pr_handle *info = (struct pr_handle *) p;

  assert (info->stack != NULL);

  return substitute_type (info, "&|");
}

/* Make a range type.  */

static boolean
pr_range_type (p, lower, upper)
     PTR p;
     bfd_signed_vma lower;
     bfd_signed_vma upper;
{
  struct pr_handle *info = (struct pr_handle *) p;
  char abl[20], abu[20];

  assert (info->stack != NULL);

  if (! substitute_type (info, ""))
    return false;

  print_vma (lower, abl, false, false);
  print_vma (upper, abu, false, false);

  return (prepend_type (info, "range (")
	  && append_type (info, "):")
	  && append_type (info, abl)
	  && append_type (info, ":")
	  && append_type (info, abu));
}

/* Make an array type.  */

/*ARGSUSED*/
static boolean
pr_array_type (p, lower, upper, stringp)
     PTR p;
     bfd_signed_vma lower;
     bfd_signed_vma upper;
     boolean stringp;
{
  struct pr_handle *info = (struct pr_handle *) p;
  char *range_type;
  char abl[20], abu[20], ab[50];

  range_type = pop_type (info);
  if (range_type == NULL)
    return false;

  if (lower == 0)
    {
      if (upper == -1)
	sprintf (ab, "|[]");
      else
	{
	  print_vma (upper + 1, abu, false, false);
	  sprintf (ab, "|[%s]", abu);
	}
    }
  else
    {
      print_vma (lower, abl, false, false);
      print_vma (upper, abu, false, false);
      sprintf (ab, "|[%s:%s]", abl, abu);
    }

  if (! substitute_type (info, ab))
    return false;

  if (strcmp (range_type, "int") != 0)
    {
      if (! append_type (info, ":")
	  || ! append_type (info, range_type))
	return false;
    }

  if (stringp)
    {
      if (! append_type (info, " /* string */"))
	return false;
    }

  return true;
}

/* Make a set type.  */

/*ARGSUSED*/
static boolean
pr_set_type (p, bitstringp)
     PTR p;
     boolean bitstringp;
{
  struct pr_handle *info = (struct pr_handle *) p;

  if (! substitute_type (info, ""))
    return false;

  if (! prepend_type (info, "set { ")
      || ! append_type (info, " }"))
    return false;

  if (bitstringp)
    {
      if (! append_type (info, "/* bitstring */"))
	return false;
    }

  return true;
}

/* Make an offset type.  */

static boolean
pr_offset_type (p)
     PTR p;
{
  struct pr_handle *info = (struct pr_handle *) p;
  char *t;

  if (! substitute_type (info, ""))
    return false;

  t = pop_type (info);
  if (t == NULL)
    return false;

  return (substitute_type (info, "")
	  && prepend_type (info, " ")
	  && prepend_type (info, t)
	  && append_type (info, "::|"));
}

/* Make a method type.  */

static boolean
pr_method_type (p, domain, argcount, varargs)
     PTR p;
     boolean domain;
     int argcount;
     boolean varargs;
{
  struct pr_handle *info = (struct pr_handle *) p;
  unsigned int len;
  char *domain_type;
  char **arg_types;
  char *s;

  len = 10;

  if (! domain)
    domain_type = NULL;
  else
    {
      if (! substitute_type (info, ""))
	return false;
      domain_type = pop_type (info);
      if (domain_type == NULL)
	return false;
      if (strncmp (domain_type, "class ", sizeof "class " - 1) == 0
	  && strchr (domain_type + sizeof "class " - 1, ' ') == NULL)
	domain_type += sizeof "class " - 1;
      else if (strncmp (domain_type, "union class ",
			sizeof "union class ") == 0
	       && (strchr (domain_type + sizeof "union class " - 1, ' ')
		   == NULL))
	domain_type += sizeof "union class " - 1;
      len += strlen (domain_type);
    }

  if (argcount <= 0)
    {
      arg_types = NULL;
      len += 15;
    }
  else
    {
      int i;

      arg_types = (char **) xmalloc (argcount * sizeof *arg_types);
      for (i = argcount - 1; i >= 0; i--)
	{
	  if (! substitute_type (info, ""))
	    return false;
	  arg_types[i] = pop_type (info);
	  if (arg_types[i] == NULL)
	    return false;
	  len += strlen (arg_types[i]) + 2;
	}
      if (varargs)
	len += 5;
    }

  /* Now the return type is on the top of the stack.  */

  s = (char *) xmalloc (len);
  if (! domain)
    *s = '\0';
  else
    strcpy (s, domain_type);
  strcat (s, "::| (");

  if (argcount < 0)
    strcat (s, "/* unknown */");
  else
    {
      int i;

      for (i = 0; i < argcount; i++)
	{
	  if (i > 0)
	    strcat (s, ", ");
	  strcat (s, arg_types[i]);
	}
      if (varargs)
	{
	  if (i > 0)
	    strcat (s, ", ");
	  strcat (s, "...");
	}
      if (argcount > 0)
	free (arg_types);
    }

  strcat (s, ")");

  if (! substitute_type (info, s))
    return false;

  free (s);

  return true;
}

/* Make a const qualified type.  */

static boolean
pr_const_type (p)
     PTR p;
{
  struct pr_handle *info = (struct pr_handle *) p;

  return substitute_type (info, "const |");
}

/* Make a volatile qualified type.  */

static boolean
pr_volatile_type (p)
     PTR p;
{
  struct pr_handle *info = (struct pr_handle *) p;

  return substitute_type (info, "volatile |");
}

/* Start accumulating a struct type.  */

static boolean
pr_start_struct_type (p, tag, id, structp, size)
     PTR p;
     const char *tag;
     unsigned int id;
     boolean structp;
     unsigned int size;
{
  struct pr_handle *info = (struct pr_handle *) p;

  info->indent += 2;

  if (! push_type (info, structp ? "struct " : "union "))
    return false;
  if (tag != NULL)
    {
      if (! append_type (info, tag))
	return false;
    }
  else
    {
      char idbuf[20];

      sprintf (idbuf, "%%anon%u", id);
      if (! append_type (info, idbuf))
	return false;
    }

  if (! append_type (info, " {"))
    return false;
  if (size != 0 || tag != NULL)
    {
      char ab[30];

      if (! append_type (info, " /*"))
	return false;

      if (size != 0)
	{
	  sprintf (ab, " size %u", size);
	  if (! append_type (info, ab))
	    return false;
	}
      if (tag != NULL)
	{
	  sprintf (ab, " id %u", id);
	  if (! append_type (info, ab))
	    return false;
	}
      if (! append_type (info, " */"))
	return false;
    }
  if (! append_type (info, "\n"))
    return false;

  info->stack->visibility = DEBUG_VISIBILITY_PUBLIC;

  return indent_type (info);
}

/* Output the visibility of a field in a struct.  */

static boolean
pr_fix_visibility (info, visibility)
     struct pr_handle *info;
     enum debug_visibility visibility;
{
  const char *s;
  char *t;
  unsigned int len;

  assert (info->stack != NULL);

  if (info->stack->visibility == visibility)
    return true;

  assert (info->stack->visibility != DEBUG_VISIBILITY_IGNORE);

  switch (visibility)
    {
    case DEBUG_VISIBILITY_PUBLIC:
      s = "public";
      break;
    case DEBUG_VISIBILITY_PRIVATE:
      s = "private";
      break;
    case DEBUG_VISIBILITY_PROTECTED:
      s = "protected";
      break;
    case DEBUG_VISIBILITY_IGNORE:
      s = "/* ignore */";
      break;
    default:
      abort ();
      return false;
    }

  /* Trim off a trailing space in the struct string, to make the
     output look a bit better, then stick on the visibility string.  */

  t = info->stack->type;
  len = strlen (t);
  assert (t[len - 1] == ' ');
  t[len - 1] = '\0';

  if (! append_type (info, s)
      || ! append_type (info, ":\n")
      || ! indent_type (info))
    return false;

  info->stack->visibility = visibility;

  return true;
}

/* Add a field to a struct type.  */

static boolean
pr_struct_field (p, name, bitpos, bitsize, visibility)
     PTR p;
     const char *name;
     bfd_vma bitpos;
     bfd_vma bitsize;
     enum debug_visibility visibility;
{
  struct pr_handle *info = (struct pr_handle *) p;
  char ab[20];
  char *t;

  if (! substitute_type (info, name))
    return false;

  if (! append_type (info, "; /* "))
    return false;

  if (bitsize != 0)
    {
      print_vma (bitsize, ab, true, false);
      if (! append_type (info, "bitsize ")
	  || ! append_type (info, ab)
	  || ! append_type (info, ", "))
	return false;
    }

  print_vma (bitpos, ab, true, false);
  if (! append_type (info, "bitpos ")
      || ! append_type (info, ab)
      || ! append_type (info, " */\n")
      || ! indent_type (info))
    return false;

  t = pop_type (info);
  if (t == NULL)
    return false;

  if (! pr_fix_visibility (info, visibility))
    return false;

  return append_type (info, t);
}

/* Finish a struct type.  */

static boolean
pr_end_struct_type (p)
     PTR p;
{
  struct pr_handle *info = (struct pr_handle *) p;
  char *s;

  assert (info->stack != NULL);
  assert (info->indent >= 2);

  info->indent -= 2;

  /* Change the trailing indentation to have a close brace.  */
  s = info->stack->type + strlen (info->stack->type) - 2;
  assert (s[0] == ' ' && s[1] == ' ' && s[2] == '\0');

  *s++ = '}';
  *s = '\0';

  return true;
}

/* Start a class type.  */

static boolean
pr_start_class_type (p, tag, id, structp, size, vptr, ownvptr)
     PTR p;
     const char *tag;
     unsigned int id;
     boolean structp;
     unsigned int size;
     boolean vptr;
     boolean ownvptr;
{
  struct pr_handle *info = (struct pr_handle *) p;
  char *tv = NULL;

  info->indent += 2;

  if (vptr && ! ownvptr)
    {
      tv = pop_type (info);
      if (tv == NULL)
	return false;
    }

  if (! push_type (info, structp ? "class " : "union class "))
    return false;
  if (tag != NULL)
    {
      if (! append_type (info, tag))
	return false;
    }
  else
    {
      char idbuf[20];

      sprintf (idbuf, "%%anon%u", id);
      if (! append_type (info, idbuf))
	return false;
    }

  if (! append_type (info, " {"))
    return false;
  if (size != 0 || vptr || ownvptr || tag != NULL)
    {
      if (! append_type (info, " /*"))
	return false;

      if (size != 0)
	{
	  char ab[20];

	  sprintf (ab, "%u", size);
	  if (! append_type (info, " size ")
	      || ! append_type (info, ab))
	    return false;
	}

      if (vptr)
	{
	  if (! append_type (info, " vtable "))
	    return false;
	  if (ownvptr)
	    {
	      if (! append_type (info, "self "))
		return false;
	    }
	  else
	    {
	      if (! append_type (info, tv)
		  || ! append_type (info, " "))
		return false;
	    }
	}

      if (tag != NULL)
	{
	  char ab[30];

	  sprintf (ab, " id %u", id);
	  if (! append_type (info, ab))
	    return false;
	}

      if (! append_type (info, " */"))
	return false;
    }

  info->stack->visibility = DEBUG_VISIBILITY_PRIVATE;

  return (append_type (info, "\n")
	  && indent_type (info));
}

/* Add a static member to a class.  */

static boolean
pr_class_static_member (p, name, physname, visibility)
     PTR p;
     const char *name;
     const char *physname;
     enum debug_visibility visibility;
{
  struct pr_handle *info = (struct pr_handle *) p;
  char *t;

  if (! substitute_type (info, name))
    return false;

  if (! prepend_type (info, "static ")
      || ! append_type (info, "; /* ")
      || ! append_type (info, physname)
      || ! append_type (info, " */\n")
      || ! indent_type (info))
    return false;

  t = pop_type (info);
  if (t == NULL)
    return false;

  if (! pr_fix_visibility (info, visibility))
    return false;

  return append_type (info, t);
}

/* Add a base class to a class.  */

static boolean
pr_class_baseclass (p, bitpos, virtual, visibility)
     PTR p;
     bfd_vma bitpos;
     boolean virtual;
     enum debug_visibility visibility;
{
  struct pr_handle *info = (struct pr_handle *) p;
  char *t;
  const char *prefix;
  char ab[20];
  char *s, *l, *n;

  assert (info->stack != NULL && info->stack->next != NULL);

  if (! substitute_type (info, ""))
    return false;

  t = pop_type (info);
  if (t == NULL)
    return false;

  if (strncmp (t, "class ", sizeof "class " - 1) == 0)
    t += sizeof "class " - 1;

  /* Push it back on to take advantage of the prepend_type and
     append_type routines.  */
  if (! push_type (info, t))
    return false;

  if (virtual)
    {
      if (! prepend_type (info, "virtual "))
	return false;
    }

  switch (visibility)
    {
    case DEBUG_VISIBILITY_PUBLIC:
      prefix = "public ";
      break;
    case DEBUG_VISIBILITY_PROTECTED:
      prefix = "protected ";
      break;
    case DEBUG_VISIBILITY_PRIVATE:
      prefix = "private ";
      break;
    default:
      prefix = "/* unknown visibility */ ";
      break;
    }

  if (! prepend_type (info, prefix))
    return false;

  if (bitpos != 0)
    {
      print_vma (bitpos, ab, true, false);
      if (! append_type (info, " /* bitpos ")
	  || ! append_type (info, ab)
	  || ! append_type (info, " */"))
	return false;
    }

  /* Now the top of the stack is something like "public A / * bitpos
     10 * /".  The next element on the stack is something like "class
     xx { / * size 8 * /\n...".  We want to substitute the top of the
     stack in before the {.  */
  s = strchr (info->stack->next->type, '{');
  assert (s != NULL);
  --s;

  /* If there is already a ':', then we already have a baseclass, and
     we must append this one after a comma.  */
  for (l = info->stack->next->type; l != s; l++)
    if (*l == ':')
      break;
  if (! prepend_type (info, l == s ? " : " : ", "))
    return false;

  t = pop_type (info);
  if (t == NULL)
    return false;

  n = (char *) xmalloc (strlen (info->stack->type) + strlen (t) + 1);
  memcpy (n, info->stack->type, s - info->stack->type);
  strcpy (n + (s - info->stack->type), t);
  strcat (n, s);

  free (info->stack->type);
  info->stack->type = n;

  free (t);

  return true;
}

/* Start adding a method to a class.  */

static boolean
pr_class_start_method (p, name)
     PTR p;
     const char *name;
{
  struct pr_handle *info = (struct pr_handle *) p;

  assert (info->stack != NULL);
  info->stack->method = name;
  return true;
}

/* Add a variant to a method.  */

static boolean
pr_class_method_variant (p, physname, visibility, constp, volatilep, voffset,
			 context)
     PTR p;
     const char *physname;
     enum debug_visibility visibility;
     boolean constp;
     boolean volatilep;
     bfd_vma voffset;
     boolean context;
{
  struct pr_handle *info = (struct pr_handle *) p;
  char *method_type;
  char *context_type;

  assert (info->stack != NULL);
  assert (info->stack->next != NULL);

  /* Put the const and volatile qualifiers on the type.  */
  if (volatilep)
    {
      if (! append_type (info, " volatile"))
	return false;
    }
  if (constp)
    {
      if (! append_type (info, " const"))
	return false;
    }

  /* Stick the name of the method into its type.  */
  if (! substitute_type (info,
			 (context
			  ? info->stack->next->next->method
			  : info->stack->next->method)))
    return false;

  /* Get the type.  */
  method_type = pop_type (info);
  if (method_type == NULL)
    return false;

  /* Pull off the context type if there is one.  */
  if (! context)
    context_type = NULL;
  else
    {
      context_type = pop_type (info);
      if (context_type == NULL)
	return false;
    }

  /* Now the top of the stack is the class.  */

  if (! pr_fix_visibility (info, visibility))
    return false;

  if (! append_type (info, method_type)
      || ! append_type (info, " /* ")
      || ! append_type (info, physname)
      || ! append_type (info, " "))
    return false;
  if (context || voffset != 0)
    {
      char ab[20];

      if (context)
	{
	  if (! append_type (info, "context ")
	      || ! append_type (info, context_type)
	      || ! append_type (info, " "))
	    return false;
	}
      print_vma (voffset, ab, true, false);
      if (! append_type (info, "voffset ")
	  || ! append_type (info, ab))
	return false;
    }

  return (append_type (info, " */;\n")
	  && indent_type (info));
}

/* Add a static variant to a method.  */

static boolean
pr_class_static_method_variant (p, physname, visibility, constp, volatilep)
     PTR p;
     const char *physname;
     enum debug_visibility visibility;
     boolean constp;
     boolean volatilep;
{
  struct pr_handle *info = (struct pr_handle *) p;
  char *method_type;

  assert (info->stack != NULL);
  assert (info->stack->next != NULL);
  assert (info->stack->next->method != NULL);

  /* Put the const and volatile qualifiers on the type.  */
  if (volatilep)
    {
      if (! append_type (info, " volatile"))
	return false;
    }
  if (constp)
    {
      if (! append_type (info, " const"))
	return false;
    }

  /* Mark it as static.  */
  if (! prepend_type (info, "static "))
    return false;

  /* Stick the name of the method into its type.  */
  if (! substitute_type (info, info->stack->next->method))
    return false;

  /* Get the type.  */
  method_type = pop_type (info);
  if (method_type == NULL)
    return false;

  /* Now the top of the stack is the class.  */

  if (! pr_fix_visibility (info, visibility))
    return false;

  return (append_type (info, method_type)
	  && append_type (info, " /* ")
	  && append_type (info, physname)
	  && append_type (info, " */;\n")
	  && indent_type (info));
}

/* Finish up a method.  */

static boolean
pr_class_end_method (p)
     PTR p;
{
  struct pr_handle *info = (struct pr_handle *) p;

  info->stack->method = NULL;
  return true;
}

/* Finish up a class.  */

static boolean
pr_end_class_type (p)
     PTR p;
{
  return pr_end_struct_type (p);
}

/* Push a type on the stack using a typedef name.  */

static boolean
pr_typedef_type (p, name)
     PTR p;
     const char *name;
{
  struct pr_handle *info = (struct pr_handle *) p;

  return push_type (info, name);
}

/* Push a type on the stack using a tag name.  */

static boolean
pr_tag_type (p, name, id, kind)
     PTR p;
     const char *name;
     unsigned int id;
     enum debug_type_kind kind;
{
  struct pr_handle *info = (struct pr_handle *) p;
  const char *t, *tag;
  char idbuf[30];

  switch (kind)
    {
    case DEBUG_KIND_STRUCT:
      t = "struct ";
      break;
    case DEBUG_KIND_UNION:
      t = "union ";
      break;
    case DEBUG_KIND_ENUM:
      t = "enum ";
      break;
    case DEBUG_KIND_CLASS:
      t = "class ";
      break;
    case DEBUG_KIND_UNION_CLASS:
      t = "union class ";
      break;
    default:
      abort ();
      return false;
    }

  if (! push_type (info, t))
    return false;
  if (name != NULL)
    tag = name;
  else
    {
      sprintf (idbuf, "%%anon%u", id);
      tag = idbuf;
    }

  if (! append_type (info, tag))
    return false;
  if (name != NULL && kind != DEBUG_KIND_ENUM)
    {
      sprintf (idbuf, " /* id %u */", id);
      if (! append_type (info, idbuf))
	return false;
    }

  return true;
}

/* Output a typedef.  */

static boolean
pr_typdef (p, name)
     PTR p;
     const char *name;
{
  struct pr_handle *info = (struct pr_handle *) p;
  char *s;

  if (! substitute_type (info, name))
    return false;

  s = pop_type (info);
  if (s == NULL)
    return false;
/*
  indent (info);
  TRACE_FPRINTF( (info->f, "typedef %s;\n", s));
*/
  free (s);

  return true;
}

/* Output a tag.  The tag should already be in the string on the
   stack, so all we have to do here is print it out.  */

/*ARGSUSED*/
static boolean
pr_tag (p, name)
     PTR p;
     const char *name;
{
  struct pr_handle *info = (struct pr_handle *) p;
  char *t;

  t = pop_type (info);
  if (t == NULL)
    return false;
/*
  indent (info);
  TRACE_FPRINTF( (info->f, "%s;\n", t));
*/
  free (t);

  return true;
}

/* Output an integer constant.  */

static boolean
pr_int_constant (p, name, val)
     PTR p;
     const char *name;
     bfd_vma val;
{
/*
  struct pr_handle *info = (struct pr_handle *) p;
  char ab[20];
  indent (info);
  print_vma (val, ab, false, false);
  TRACE_FPRINTF( (info->f, "const int %s = %s;\n", name, ab));
 */
  return true;
}

/* Output a floating point constant.  */

static boolean
pr_float_constant (p, name, val)
     PTR p;
     const char *name;
     double val;
{
/*
  struct pr_handle *info = (struct pr_handle *) p;
  indent (info);
  TRACE_FPRINTF( (info->f, "const double %s = %g;\n", name, val));
 */
  return true;
}

/* Output a typed constant.  */

static boolean
pr_typed_constant (p, name, val)
     PTR p;
     const char *name;
     bfd_vma val;
{
  struct pr_handle *info = (struct pr_handle *) p;
  char *t;

  t = pop_type (info);
  if (t == NULL)
    return false;
/*
  char ab[20];
  indent (info);
  print_vma (val, ab, false, false);
  TRACE_FPRINTF( (info->f, "const %s %s = %s;\n", t, name, ab));
*/
  free (t);

  return true;
}

/* Output a variable.  */

static boolean
pr_variable (p, name, kind, val)
     PTR p;
     const char *name;
     enum debug_var_kind kind;
     bfd_vma val;
{
  struct pr_handle *info = (struct pr_handle *) p;
  char *t;
  char ab[20];
  (void)ab;

  if (! substitute_type (info, name))
    return false;

  t = pop_type (info);
  if (t == NULL)
    return false;

#if 0
  indent (info);
  switch (kind)
    {
    case DEBUG_STATIC:
    case DEBUG_LOCAL_STATIC:
      TRACE_FPRINTF( (info->f, "static "));
      break;
    case DEBUG_REGISTER:
      TRACE_FPRINTF( (info->f, "register "));
      break;
    default:
      break;
    }
  print_vma (val, ab, true, true);
  TRACE_FPRINTF( (info->f, "%s /* %s */;\n", t, ab));
#else /* 0 */
#if 0
	if (kind==DEBUG_STATIC || kind==DEBUG_LOCAL_STATIC) {
		print_vma (val, ab, true, true);
		TRACE_FPRINTF( (info->f, "STATIC_VAR: %s /* %s */;\n", t, ab));
	}
#endif /* 0 */
#endif /* !0 */

  free (t);

  return true;
}

/* Start outputting a function.  */

static boolean
pr_start_function (p, name, global)
     PTR p;
     const char *name;
     boolean global;
{
  struct pr_handle *info = (struct pr_handle *) p;
  char *t;

  if (! substitute_type (info, name))
    return false;

  t = pop_type (info);
  if (t == NULL)
    return false;

#if 0
  indent (info);
  if (! global)
    TRACE_FPRINTF( (info->f, "static "));
  TRACE_FPRINTF( (info->f, "%s (", t));
  info->parameter = 1;
#else /* 0 */
	if (info->functions_size==info->functions_maxsize) {
		info->functions_maxsize *= 2;
		info->functions = xrealloc(info->functions,
			info->functions_maxsize*sizeof(debug_function_t));
		assert(info->functions!=0);
	}
	/* info->functions[info->functions_size] = xmalloc(sizeof(debug_function_t)); */
	info->function = &info->functions[info->functions_size];
	++info->functions_size;
	info->function->symbol = NULL;
	info->function->lines = NULL;
	info->function->lines_count = 0;
	info->function->max_lines_count = 0;
	info->function->name = t;
	info->function->filename = NULL;
	info->function->block = NULL;
	info->function->argv = NULL;
	info->function->argc = 0;
	info->function->max_argc = 0;
#endif /* !0 */
	return true;
}

/* Output a function parameter.  */

static boolean
pr_function_parameter (p, name, kind, val)
     PTR p;
     const char *name;
     enum debug_parm_kind kind;
     bfd_vma val;
{
  struct pr_handle *info = (struct pr_handle *) p;
  debug_function_t*	f = info->function;
  char *t;
  char ab[20];
  (void)ab;

  if (kind == DEBUG_PARM_REFERENCE
      || kind == DEBUG_PARM_REF_REG)
    {
      if (! pr_reference_type (p))
	return false;
    }

  if (! substitute_type (info, name))
    return false;

  t = pop_type (info);
  if (t == NULL)
    return false;

#if 0
  if (info->parameter != 1)
    TRACE_FPRINTF( (info->f, ", "));

  if (kind == DEBUG_PARM_REG || kind == DEBUG_PARM_REF_REG)
    TRACE_FPRINTF( (info->f, "register "));

  print_vma (val, ab, true, true);
  TRACE_FPRINTF( (info->f, "%s /* %s */", t, ab));
  free (t);
  ++info->parameter;
#else /* 0 */
	assert(f!=NULL);
	if (f->argv==NULL) {
		f->max_argc = 7; /* rarely anyone has more than that many args... */
		f->argv = xmalloc(sizeof(debug_parameter_t)*f->max_argc);
	} else if (f->argc==f->max_argc) {
		f->max_argc *= 2;
		f->argv = realloc(f->argv,sizeof(debug_parameter_t)*f->max_argc);
	}
	f->argv[f->argc].offset = val;
	f->argv[f->argc].name = t;
	++f->argc;
#endif /* !0 */
	return true;
}

/* Start writing out a block.  */

static boolean
pr_start_block (p, addr)
     PTR p;
     bfd_vma addr;
{
	struct pr_handle *info = (struct pr_handle *) p;
	char ab[20];
	debug_block_t*	block = 0;
	(void)ab;
#if 0
  if (info->parameter > 0)
    {
      TRACE_FPRINTF( (info->f, ")\n"));
      info->parameter = 0;
    }
  indent (info);
  print_vma (addr, ab, true, true);
  TRACE_FPRINTF( (info->f, "{ /* %s */\n", ab));
  info->indent += 2;
#else
	if (info->block) {
		if (info->block->childs_count==0)
			info->block->childs = xmalloc(sizeof(debug_block_t));
		else
			info->block->childs = xrealloc(info->block->childs,
						info->block->childs_count*sizeof(debug_block_t));
		block = &info->block->childs[info->block->childs_count];
	} else {
		block = xmalloc(sizeof(debug_block_t));
		info->function->block = block;
	}
	block->begin_addr = addr;
	block->end_addr = 0;
	block->parent = info->block;
	block->childs = NULL;
	block->childs_count = 0;
	info->block = block;
#endif
	return true;
}

/* Write out line number information.  */

static boolean
pr_lineno (p, filename, lineno, addr)
     PTR p;
     const char *filename;
     unsigned long lineno;
     bfd_vma addr;
{
	struct pr_handle *info = (struct pr_handle *) p;
	char ab[20];
	debug_function_t*	f = info->function;
	(void)ab;

#if 0
  indent (info);
  print_vma (addr, ab, true, true);
  TRACE_FPRINTF( (info->f, "/* file %s line %lu addr %s */\n", filename, lineno, ab));
#else /* 0 */
	if (f==NULL)	/* FIXME: skips junk silently. */
		return true;
	/* assert(f!=NULL); */
	if (f->filename==NULL) {
		f->filename = filename;
		assert(f->lines==0);
		f->max_lines_count = 4;
		f->lines = xmalloc(sizeof(debug_lineno_t)*f->max_lines_count);
	}
	if (f->lines_count==f->max_lines_count) {
		f->max_lines_count *= 2;
		f->lines = xrealloc(f->lines, sizeof(debug_lineno_t)*f->max_lines_count);
	}
	f->lines[f->lines_count].lineno = lineno;
	f->lines[f->lines_count].addr = addr;
	++f->lines_count;
#endif /* !0 */

  return true;
}

/* Finish writing out a block.  */

static boolean
pr_end_block (p, addr)
     PTR p;
     bfd_vma addr;
{
	struct pr_handle *info = (struct pr_handle *) p;

#if 0
  char ab[20];

  info->indent -= 2;
  indent (info);
  print_vma (addr, ab, true, true);
  TRACE_FPRINTF( (info->f, "} /* %s */\n", ab));
#else /* 0 */
	assert(info->block!=0);
	info->block->end_addr = addr;
	info->block = info->block->parent;
#endif /* !0 */

	return true;
}

/* Finish writing out a function.  */

/*ARGSUSED*/
static boolean
pr_end_function (p)
     PTR p;
{
	struct pr_handle *info = (struct pr_handle *) p;
	assert(info->block==0);
	info->function = NULL;
	return true;
}

/* third parameter to segv_action. */
/* Got it after a bit of head scratching and stack dumping. */
typedef struct {
	u_int32_t	foo1;	/* +0x00 */
	u_int32_t	foo2;
	u_int32_t	foo3;
	u_int32_t	foo4;	/* usually 2 */
	u_int32_t	foo5;	/* +0x10 */
	u_int32_t	xgs;	/* always zero */
	u_int32_t	xfs;	/* always zero */
	u_int32_t	xes;	/* always es=ds=ss */
	u_int32_t	xds;	/* +0x20 */
	u_int32_t	edi;
	u_int32_t	esi;
	u_int32_t	ebp;
	u_int32_t	esp;	/* +0x30 */
	u_int32_t	ebx;
	u_int32_t	edx;
	u_int32_t	ecx;
	u_int32_t	eax;	/* +0x40 */
	u_int32_t	foo11;	/* usually 0xe */
	u_int32_t	foo12;	/* usually 0x6 */
	u_int32_t	eip;	/* instruction pointer */
	u_int32_t	xcs;	/* +0x50 */
	u_int32_t	foo21;	/* usually 0x2 */
	u_int32_t	foo22;	/* second stack pointer?! Probably. */
	u_int32_t	xss;
	u_int32_t	foo31;	/* +0x60 */ /* usually 0x0 */
	u_int32_t	foo32;	/* usually 0x2 */
	u_int32_t	fault_addr;	/* Address which caused a fault */
	u_int32_t	foo41;	/* usually 0x2 */
} signal_regs_t;

signal_regs_t*	ptrace_regs = 0;	/* Tells my_ptrace to "ptrace" current process" */
/*
 * my_ptrace: small wrapper around ptrace.
 * Act as normal ptrace if ptrace_regs==0.
 * Read data from current process if ptrace_regs!=0.
 */
static int
my_ptrace(	int	request,
		int	pid,
		int	addr,
		int	data)
{
	if (ptrace_regs==0)
		return ptrace(request, pid, addr, data);
	/* we are tracing ourselves! */
	switch (request) {
	case PTRACE_ATTACH:	return 0;
	case PTRACE_CONT:	return 0;
	case PTRACE_DETACH:	return 0;
	case PTRACE_PEEKUSER:
				switch (addr / 4) {
				case EIP:	return ptrace_regs->eip;
				case EBP:	return ptrace_regs->ebp;
				default:	assert(0);
				}
	case PTRACE_PEEKTEXT:	/* FALLTHROUGH */
	case PTRACE_PEEKDATA:	return *(int*)(addr);
	default:		assert(0);
	}
	errno = 1;	/* what to do here? */
	return 1;	/* failed?! */
}

#define	MAXARGS	6

/*
 * To minimize the number of parameters.
 */
typedef struct {
	asymbol**		syms;	/* Sorted! */
	int			symcount;
	debug_function_t**	functions;
	int			functions_size;
} symbol_data_t;

/*
 * Perform a search. A binary search for a symbol.
 */
static void
decode_symbol(	symbol_data_t*		symbol_data,
		const unsigned long	addr,
		char*			buf,
		const int		bufsize)
{
	asymbol**	syms = symbol_data->syms;
	const int	symcount = symbol_data->symcount;
	int		bottom = 0;
	int		top = symcount - 1;
	int		i;
	if (symcount==0) {
		sprintf(buf, "????");
		return;
	}
	while (top>bottom+1) {
		i = (top+bottom) / 2;
		if (bfd_asymbol_value(syms[i])==addr) {
			sprintf(buf, "%s", syms[i]->name);
			return;
		} else if (bfd_asymbol_value(syms[i]) > addr)
			top = i;
		else
			bottom = i;
	}
	i = bottom;
	if (addr<bfd_asymbol_value(syms[i]) || addr>(syms[i]->section->vma+syms[i]->section->_cooked_size))
		sprintf(buf, "????");
	else
		sprintf(buf, "%s + 0x%lx", syms[i]->name, addr-bfd_asymbol_value(syms[i]));
}

/*
 * 1. Perform a binary search for an debug_function_t.
 * 2. Fill buf/bufsize with name, parameters and lineno, if found
 *    Or with '????' otherwise.
 */
static debug_function_t*
find_debug_function_t(	symbol_data_t*		symbol_data,
			const pid_t		pid,
			const unsigned long	fp,	/* frame pointer */
			const unsigned long	addr,
			char*			buf,	/* string buffer */
			const int		bufsize)/* FIXME: not used! */
{
	debug_function_t**	syms = symbol_data->functions;
	debug_function_t*	f = NULL;
	debug_block_t*		block = NULL;
	debug_lineno_t*		lineno = NULL;
	const int		symcount = symbol_data->functions_size;
	int			bottom = 0;
	int			top = symcount - 1;
	int			i;
	char*			bufptr = buf;

	if (symcount==0) {
		sprintf(buf, "????");
		return NULL;
	}
	while (top>bottom+1) {
		i = (top+bottom) / 2;
		if (syms[i]->block->begin_addr==addr) {
			f = syms[i];
			break;
		} else if (syms[i]->block->begin_addr > addr)
				top = i;
		else
			if (syms[i]->block->end_addr >= addr) {
				f = syms[i];
				break;
			} else
				bottom = i;
	}
	i = bottom;
	if (f!=0)
		block = f->block;
	else {
		block = syms[i]->block;
		if (block->begin_addr>=addr && block->end_addr<=addr)
			f = syms[i];
	}
	if (f==0)
		sprintf(buf, "????");
	else {
		/*
		 * Do the backtrace the GDB way...
		 */
		unsigned long	arg;
		/* assert(f->lines_count>0); */
		if (f->lines_count>0) {
			lineno = &f->lines[f->lines_count-1];
			for (i=1; i<f->lines_count; ++i)
				if (f->lines[i].addr>addr) {
					lineno = &f->lines[i-1];
					break;
				}
		}
		bufptr[0] = 0;
		bufptr += sprintf(bufptr, "%s+0x%lx (", f->name, addr-block->begin_addr);
		for (i=0; i<f->argc; ++i) {
			bufptr += sprintf(bufptr, "%s = ", f->argv[i].name);
			/* FIXME: better parameter printing */
			errno = 0;
			arg = my_ptrace(PTRACE_PEEKDATA, pid, fp+f->argv[i].offset, 0);
			assert(errno==0);
			bufptr += sprintf(bufptr, "0x%x", arg);
			if (i!=f->argc-1)
				bufptr += sprintf(bufptr, ", ");
		}
		if (lineno!=0)
			bufptr += sprintf(bufptr, ") at %s:%d", f->filename, lineno->lineno);
	}
	return f;
}

/*
 * Advance through the stacks and display frames as needed.
 */
static int
my_crawl(	int		pid,
		symbol_data_t*	symbol_data,
		int		fout)
{
	unsigned long		pc = 0;
	unsigned long		fp = 0;
	unsigned long		nextfp;
	unsigned long		nargs;
	unsigned long		i;
	unsigned long		arg;
	char			buf[8096];	// FIXME: enough?
	debug_function_t*	f = 0;

	errno = 0;

	pc = my_ptrace(PTRACE_PEEKUSER, pid, EIP * 4, 0);
	if (!errno)
		fp = my_ptrace(PTRACE_PEEKUSER, pid, EBP * 4, 0);

	if (!errno) {
#if 1
		f = find_debug_function_t(symbol_data, pid, fp, pc, buf, sizeof(buf));
		fdprintf(fout,"0x%08lx: %s", pc, buf);
		for ( ; !errno && fp; ) {
			nextfp = my_ptrace(PTRACE_PEEKDATA, pid, fp, 0);
			if (errno)
				break;

			if (f==0) {
				nargs = (nextfp - fp - 8) / 4;
				if (nargs > MAXARGS)
					nargs = MAXARGS;
				if (nargs > 0) {
					fdputs(" (", fout);
					for (i = 1; i <= nargs; i++) {
						arg = my_ptrace(PTRACE_PEEKDATA, pid, fp + 4 * (i + 1), 0);
						if (errno)
							break;
						fdprintf(fout,"%lx", arg);
						if (i < nargs)
							fdputs(", ", fout);
					}
					fdputc(')', fout);
					nargs = nextfp - fp - 8 - (4 * nargs);
					if (!errno && nargs > 0)
						fdprintf(fout," + %lx\n", nargs);
					else
						fdputc('\n', fout);
				} else
					fdputc('\n', fout);
			} else
				fdputc('\n', fout);

			if (errno || !nextfp)
				break;
			pc = my_ptrace(PTRACE_PEEKDATA, pid, fp + 4, 0);
			fp = nextfp;
			if (errno)
				break;
			f = find_debug_function_t(symbol_data, pid, fp, pc, buf, sizeof(buf));
			fdprintf(fout,"0x%08lx: %s", pc, buf);
		}
#else /* 1 */
		decode_symbol(symbol_data, pc, buf, sizeof(buf));
		fdprintf(fout,"0x%08lx: %s", pc, buf);
		for ( ; !errno && fp; ) {
			nextfp = my_ptrace(PTRACE_PEEKDATA, pid, fp, 0);
			if (errno)
				break;

			nargs = (nextfp - fp - 8) / 4;
			if (nargs > MAXARGS)
				nargs = MAXARGS;
			if (nargs > 0) {
				fputs(" (", fout);
				for (i = 1; i <= nargs; i++) {
					arg = my_ptrace(PTRACE_PEEKDATA, pid, fp + 4 * (i + 1), 0);
					if (errno)
						break;
					fdprintf(fout,"%lx", arg);
					if (i < nargs)
						fputs(", ", fout);
				}
				fdputc(')', fout);
				nargs = nextfp - fp - 8 - (4 * nargs);
				if (!errno && nargs > 0)
					fdprintf(fout," + %lx\n", nargs);
				else
					fdputc('\n', fout);
			} else
				fdputc('\n', fout);

			if (errno || !nextfp)
				break;
			pc = my_ptrace(PTRACE_PEEKDATA, pid, fp + 4, 0);
			fp = nextfp;
			if (errno)
				break;
			decode_symbol(symbol_data, pc, buf, sizeof(buf));
			fdprintf(fout,"0x%08lx: %s", pc, buf);
		}
#endif /* !1 */
	}
	if (errno)
		perror("my_crawl");
	return errno;
}

/* layout from /usr/src/linux/arch/i386/kernel/process.c */
static void
show_regs(	signal_regs_t*	regs,
		int		fd)
{
	/* long cr0 = 0L, cr2 = 0L, cr3 = 0L; */

	fdprintf(fd,"\n");
	fdprintf(fd,"FAULT ADDR: %08x\n", regs->fault_addr);
	fdprintf(fd,"EIP: %04x:[<%08x>]",0xffff & regs->xcs,regs->eip);
	if (regs->xcs & 3)
		fdprintf(fd," ESP: %04x:%08x",0xffff & regs->xss,regs->esp);
	/*fdprintf(fd," EFLAGS: %08lx\n",regs->eflags); */
	fdprintf(fd, "\n");
	fdprintf(fd,"EAX: %08x EBX: %08x ECX: %08x EDX: %08x\n",
		regs->eax,regs->ebx,regs->ecx,regs->edx);
	fdprintf(fd,"ESI: %08x EDI: %08x EBP: %08x",
		regs->esi, regs->edi, regs->ebp);
	fdprintf(fd," DS: %04x ES: %04x\n",
		0xffff & regs->xds,0xffff & regs->xes);
	/*
	__asm__("movl %%cr0, %0": "=r" (cr0));
	__asm__("movl %%cr2, %0": "=r" (cr2));
	__asm__("movl %%cr3, %0": "=r" (cr3));
	fprintf(stderr,"CR0: %08lx CR2: %08lx CR3: %08lx\n", cr0, cr2, cr3); */
}

/*
 * Load a BFD for an executable based on PID. Return 0 on failure.
 */
static bfd*
load_bfd(	const int	pid)
{
	char	filename[512];
	bfd*	abfd = 0;

	/* Get the contents from procfs. */
#if 1
	sprintf(filename, "/proc/%d/exe", pid);
#else
	sprintf(filename, "crashing");
#endif 

	if ((abfd = bfd_openr (filename, 0))== NULL)
		bfd_nonfatal (filename);
	else {
		char**	matching;
		assert(bfd_check_format(abfd, bfd_archive)!=true);

		/*
		 * There is no indication in BFD documentation that it should be done.
		 * God knows why...
		 */
		if (!bfd_check_format_matches (abfd, bfd_object, &matching)) {
			bfd_nonfatal (bfd_get_filename (abfd));
			if (bfd_get_error () == bfd_error_file_ambiguously_recognized) {
				list_matching_formats (matching);
				free (matching);
			}
		}
	}
	return abfd;
}

/*
 * Those are for qsort. We need only function addresses, so all the others don't count.
 */
/*
 * Compare two BFD::asymbol-s.
 */
static int 
compare_symbols(const void*	ap,
		const void*	bp)
{
	const asymbol *a = *(const asymbol **)ap;
	const asymbol *b = *(const asymbol **)bp;
	if (bfd_asymbol_value (a) > bfd_asymbol_value (b))
		return 1;
	else if (bfd_asymbol_value (a) < bfd_asymbol_value (b))
		return -1;
	return 0;
}

/*
 * Compare two debug_asymbol_t-s.
 */
static int 
compare_debug_function_t(const void*	ap,
			const void*	bp)
{
	const debug_function_t *a = *(const debug_function_t **)ap;
	const debug_function_t *b = *(const debug_function_t **)bp;
	assert(a->block!=0);
	assert(b->block!=0);
	{
		const bfd_vma	addr1 = a->block->begin_addr;
		const bfd_vma	addr2 = b->block->begin_addr;
		if (addr1 > addr2)
			return 1;
		else if (addr2 > addr1)
			return -1;
	}
	return 0;
}

/*
 * Filter out (in place) symbols that are useless for stack tracing.
 * COUNT is the number of elements in SYMBOLS.
 * Return the number of useful symbols.
 */

static long
remove_useless_symbols(	asymbol**	symbols,
			long		count)
{
	asymbol**	in_ptr = symbols;
	asymbol**	out_ptr = symbols;

	while (--count >= 0) {
		asymbol *sym = *in_ptr++;

		if (sym->name == NULL || sym->name[0] == '\0' || sym->value==0)
			continue;
		if (sym->flags & (BSF_DEBUGGING))
			continue;
		if (bfd_is_und_section (sym->section) || bfd_is_com_section (sym->section))
			continue;
		*out_ptr++ = sym;
	}
	return out_ptr - symbols;
}

/*
 * Debugging information.
 */
static	bfd*			abfd = 0;
static	PTR			dhandle = 0;
static	asymbol**		syms = 0;
static	long			symcount = 0;
static	asymbol**		sorted_syms = 0;
static	long			sorted_symcount = 0;
static	debug_function_t**	functions = 0;
static	int			functions_size = 0;
static	int			sigreport = SIGUSR1;
static	pthread_t		segv_tid;	/* What thread did SEGV? */
static	pid_t			segv_pid;

/*
 * We'll get here after a SIGSEGV. But you can install it on other signals, too :)
 * Because we are in the middle of the SIGSEGV, we are on our own. We can't do
 * any malloc(), any fopen(), nothing. The last is actually a sin. We event can't
 * fprintf(stderr,...)!!!
 */
static void
segv_action(int signo, siginfo_t* siginfo, void* ptr)
{
	symbol_data_t		symbol_data;
	int			fd = -1;

	segv_pid = getpid();
	segv_tid = pthread_self();
	fd = open_log_file(segv_tid, segv_pid);
	/* signal(SIGSEGV, SIG_DFL); */
	ptrace_regs = (signal_regs_t*)ptr;
	assert(ptrace_regs!=0);

	/* Show user how guilty we are. */
	fdprintf(fd,"--------- SEGV in PROCESS %d, THREAD %d ---------------\n", segv_pid, pthread_self());
	show_regs(ptrace_regs, fd);

	/* Some form of stack trace, too. */
	fdprintf(fd, "STACK TRACE:\n");

	symbol_data.syms = sorted_syms;
	symbol_data.symcount = sorted_symcount;
	symbol_data.functions = functions;
	symbol_data.functions_size = functions_size;
	my_crawl(segv_pid, &symbol_data, fd);
	//fflush(stdout);
	close(fd);
	linuxthreads_notify_others(sigreport);
}


static void
report_action(int signo, siginfo_t* siginfo, void* ptr)
{
	const int	pid = getpid();
	pthread_t	tid = pthread_self();
	symbol_data_t	symbol_data;
	int		fd;
	if (pthread_equal(tid, segv_tid)) {
		/* We have already printed our stack trace... */
		return;
	}

	fd = open_log_file(tid, pid);
	fdprintf(fd, "REPORT: CURRENT PROCESS:%d, THREAD:%d\n", getpid(), pthread_self());
	/* signal(SIGSEGV, SIG_DFL); */
	ptrace_regs = (signal_regs_t*)ptr;
	assert(ptrace_regs!=0);

	/* Show user how guilty we are. */
	fdprintf(fd,"--------- STACK TRACE FOR PROCESS %d, THREAD %d ---------------\n", pid, pthread_self());
	show_regs(ptrace_regs, fd);

	/* Some form of stack trace, too. */
	fdprintf(fd, "STACK TRACE:\n");

	symbol_data.syms = sorted_syms;
	symbol_data.symcount = sorted_symcount;
	symbol_data.functions = functions;
	symbol_data.functions_size = functions_size;
	my_crawl(pid, &symbol_data, fd);
	//fflush(stdout);
	close(fd);
	/* Tell segv_thread to proceed after pause(). */
	/*pthread_kill(segv_tid, sigreport);
	kill(segv_pid, sigreport);
	pthread_cancel(tid); */
}

/*
 * Main library routine. Just call it on your program.
 */
int
pstack_install_segv_action(	const char*	path_format_)
{
	const int		pid = getpid();
	struct sigaction	act;

	/* Store what we have to for later usage. */
	path_format = path_format_;

	/* We need a signal action for SIGSEGV and sigreport ! */
	sigreport = SIGUSR1;
	act.sa_handler = 0;
	sigemptyset(&act.sa_mask);
	act.sa_flags = SA_SIGINFO|SA_ONESHOT;	/* Just one SIGSEGV. */
	act.sa_sigaction = segv_action;
	act.sa_restorer = NULL;
	if (sigaction(SIGSEGV, &act, NULL)!=0) {
		perror("sigaction");
		return 1;
	}
	act.sa_sigaction = report_action;
	act.sa_flags = SA_SIGINFO;		/* But many sigreports. */
	if (sigaction(sigreport, &act, NULL)!=0) {
		perror("sigaction");
		return 1;
	}

	/* And a little setup for libiberty. */
	program_name = "crashing";
	xmalloc_set_program_name (program_name);

	/* Umm, and initialize BFD, too */
	bfd_init();
#if 0
	list_supported_targets(0, stdout);
	set_default_bfd_target(); 
#endif /* 0 */

	if ((abfd = load_bfd(pid))==0)
		fprintf(stderr, "BFD load failed..\n");
	else {
		long    storage_needed= (bfd_get_file_flags(abfd) & HAS_SYMS) ?
		  bfd_get_symtab_upper_bound (abfd) : 0;
		long	i;
		(void)i;

		if (storage_needed < 0)
			fprintf(stderr, "Symbol table size estimation failure.\n");
		else if (storage_needed > 0) {
			syms = (asymbol **) xmalloc (storage_needed);
			symcount = bfd_canonicalize_symtab (abfd, syms);

			TRACE_FPRINTF((stderr, "TOTAL: %ld SYMBOLS.\n", symcount));
			/* We need debugging info, too! */
			if (symcount==0 || (dhandle = read_debugging_info (abfd, syms, symcount))==0)
				fprintf(stderr, "NO DEBUGGING INFORMATION FOUND.\n");

			/* We make a copy of syms to sort.  We don't want to sort syms
			because that will screw up the relocs.  */
			sorted_syms = (asymbol **) xmalloc (symcount * sizeof (asymbol *));
			memcpy (sorted_syms, syms, symcount * sizeof (asymbol *));

#if 0
			for (i=0; i<symcount; ++i)
				if (syms[i]->name!=0 && strlen(syms[i]->name)>0 && syms[i]->value!=0)
					printf("%08lx T %s\n", syms[i]->section->vma + syms[i]->value, syms[i]->name);
#endif
			sorted_symcount = remove_useless_symbols (sorted_syms, symcount);
			TRACE_FPRINTF((stderr, "SORTED: %ld SYMBOLS.\n", sorted_symcount));

			/* Sort the symbols into section and symbol order */
			qsort (sorted_syms, sorted_symcount, sizeof (asymbol *), compare_symbols);
#if 0
			for (i=0; i<sorted_symcount; ++i)
				if (sorted_syms[i]->name!=0 && strlen(sorted_syms[i]->name)>0 && sorted_syms[i]->value!=0)
					printf("%08lx T %s\n", sorted_syms[i]->section->vma + sorted_syms[i]->value, sorted_syms[i]->name);
#endif
			/* We have symbols, we need debugging info somehow sorted out. */
			if (dhandle==0) {
				fprintf(stderr, "STACK TRACE WILL BE UNCOMFORTABLE.\n");
			} else {
				/* Start collecting the debugging information.... */
				struct pr_handle info;

				info.f = stdout;
				info.indent = 0;
				info.stack = NULL;
				info.parameter = 0;
				info.block = NULL;
				info.function = NULL;
				info.functions_size = 0;
				info.functions_maxsize = 1000;
				info.functions = (debug_function_t*)xmalloc(sizeof(debug_function_t)*info.functions_maxsize);
				debug_write (dhandle, &pr_fns, (PTR) &info);
				TRACE_FPRINTF((stdout, "\n%d DEBUG SYMBOLS\n", info.functions_size));
				assert(info.functions_size!=0);
				functions = xmalloc(sizeof(debug_function_t*)*info.functions_size);
				functions_size = info.functions_size;
				for (i=0; i<functions_size; ++i)
					functions[i] = &info.functions[i];
				/* Sort the symbols into section and symbol order */
				qsort (functions, functions_size, sizeof(debug_function_t*),
					compare_debug_function_t);
#if 0
				for (i=0; i<info.functions_size; ++i)
					fprintf(stdout, "%08lx T %s\n", info.functions[i].block->begin_addr, info.functions[i].name);
#endif
				fflush(stdout);
			}
		} else /* storage_needed == 0 */
			fprintf(stderr, "NO SYMBOLS FOUND.\n");
	}
	return 0;
}

/*********************************************************************/
/*********************************************************************/
/*********************************************************************/