corefile.c

来自「基于4个mips核的noc设计」· C语言 代码 · 共 742 行 · 第 1/2 页

C
742
字号
/* corefile.c   Copyright 2000, 2001 Free Software Foundation, Inc.   This file is part of GNU Binutils.   This program 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.  */#include "libiberty.h"#include "gprof.h"#include "corefile.h"#include "symtab.h"bfd *core_bfd;int core_num_syms;asymbol **core_syms;asection *core_text_sect;PTR core_text_space;int min_insn_size;int offset_to_code;/* For mapping symbols to specific .o files during file ordering.  */struct function_map{  char *function_name;  char *file_name;};struct function_map *symbol_map;unsigned int symbol_map_count;extern void i386_find_call  PARAMS ((Sym *, bfd_vma, bfd_vma));extern void alpha_find_call PARAMS ((Sym *, bfd_vma, bfd_vma));extern void vax_find_call   PARAMS ((Sym *, bfd_vma, bfd_vma));extern void tahoe_find_call PARAMS ((Sym *, bfd_vma, bfd_vma));extern void sparc_find_call PARAMS ((Sym *, bfd_vma, bfd_vma));static voidDEFUN (read_function_mappings, (filename), const char *filename){  FILE *file = fopen (filename, "r");  char dummy[1024];  int count = 0;  if (!file)    {      fprintf (stderr, _("%s: could not open %s.\n"), whoami, filename);      done (1);    }  /* First parse the mapping file so we know how big we need to     make our tables.  We also do some sanity checks at this     time.  */  while (!feof (file))    {      int matches;      matches = fscanf (file, "%[^\n:]", dummy);      if (!matches)	{	  fprintf (stderr, _("%s: unable to parse mapping file %s.\n"),		   whoami, filename);	  done (1);	}      /* Just skip messages about files with no symbols.  */      if (!strncmp (dummy, "No symbols in ", 14))	{	  fscanf (file, "\n");	  continue;	}      /* Don't care what else is on this line at this point.  */      fscanf (file, "%[^\n]\n", dummy);      count++;    }  /* Now we know how big we need to make our table.  */  symbol_map = ((struct function_map *)		xmalloc (count * sizeof (struct function_map)));  /* Rewind the input file so we can read it again.  */  rewind (file);  /* Read each entry and put it into the table.  */  count = 0;  while (!feof (file))    {      int matches;      char *tmp;      matches = fscanf (file, "%[^\n:]", dummy);      if (!matches)	{	  fprintf (stderr, _("%s: unable to parse mapping file %s.\n"),		   whoami, filename);	  done (1);	}      /* Just skip messages about files with no symbols.  */      if (!strncmp (dummy, "No symbols in ", 14))	{	  fscanf (file, "\n");	  continue;	}      /* dummy has the filename, go ahead and copy it.  */      symbol_map[count].file_name = xmalloc (strlen (dummy) + 1);      strcpy (symbol_map[count].file_name, dummy);      /* Now we need the function name.  */      fscanf (file, "%[^\n]\n", dummy);      tmp = strrchr (dummy, ' ') + 1;      symbol_map[count].function_name = xmalloc (strlen (tmp) + 1);      strcpy (symbol_map[count].function_name, tmp);      count++;    }  /* Record the size of the map table for future reference.  */  symbol_map_count = count;}voidDEFUN (core_init, (a_out_name), const char *a_out_name){  core_bfd = bfd_openr (a_out_name, 0);  if (!core_bfd)    {      perror (a_out_name);      done (1);    }  if (!bfd_check_format (core_bfd, bfd_object))    {      fprintf (stderr, _("%s: %s: not in a.out format\n"), whoami, a_out_name);      done (1);    }  /* Get core's text section.  */  core_text_sect = bfd_get_section_by_name (core_bfd, ".text");  if (!core_text_sect)    {      core_text_sect = bfd_get_section_by_name (core_bfd, "$CODE$");      if (!core_text_sect)	{	  fprintf (stderr, _("%s: can't find .text section in %s\n"),		   whoami, a_out_name);	  done (1);	}    }  /* Read core's symbol table.  */  /* This will probably give us more than we need, but that's ok.  */  core_num_syms = bfd_get_symtab_upper_bound (core_bfd);  if (core_num_syms < 0)    {      fprintf (stderr, "%s: %s: %s\n", whoami, a_out_name,	       bfd_errmsg (bfd_get_error ()));      done (1);    }  core_syms = (asymbol **) xmalloc (core_num_syms);  core_num_syms = bfd_canonicalize_symtab (core_bfd, core_syms);  if (core_num_syms < 0)    {      fprintf (stderr, "%s: %s: %s\n", whoami, a_out_name,	       bfd_errmsg (bfd_get_error ()));      done (1);    }  min_insn_size = 1;  offset_to_code = 0;  switch (bfd_get_arch (core_bfd))    {    case bfd_arch_vax:    case bfd_arch_tahoe:      offset_to_code = 2;      break;    case bfd_arch_alpha:      min_insn_size = 4;      break;    default:      break;    }  if (function_mapping_file)    read_function_mappings (function_mapping_file);}/* Read in the text space of an a.out file.  */voidDEFUN (core_get_text_space, (core_bfd), bfd * core_bfd){  core_text_space = (PTR) malloc (core_text_sect->_raw_size);  if (!core_text_space)    {      fprintf (stderr, _("%s: ran out room for %lu bytes of text space\n"),	       whoami, (unsigned long) core_text_sect->_raw_size);      done (1);    }  if (!bfd_get_section_contents (core_bfd, core_text_sect, core_text_space,				 0, core_text_sect->_raw_size))    {      bfd_perror ("bfd_get_section_contents");      free (core_text_space);      core_text_space = 0;    }  if (!core_text_space)    fprintf (stderr, _("%s: can't do -c\n"), whoami);}voidDEFUN (find_call, (parent, p_lowpc, p_highpc),       Sym * parent AND bfd_vma p_lowpc AND bfd_vma p_highpc){  switch (bfd_get_arch (core_bfd))    {    case bfd_arch_i386:      i386_find_call (parent, p_lowpc, p_highpc);      break;    case bfd_arch_alpha:      alpha_find_call (parent, p_lowpc, p_highpc);      break;    case bfd_arch_vax:      vax_find_call (parent, p_lowpc, p_highpc);      break;    case bfd_arch_sparc:      sparc_find_call (parent, p_lowpc, p_highpc);      break;    case bfd_arch_tahoe:      tahoe_find_call (parent, p_lowpc, p_highpc);      break;    default:      fprintf (stderr, _("%s: -c not supported on architecture %s\n"),	       whoami, bfd_printable_name(core_bfd));      /* Don't give the error more than once.  */      ignore_direct_calls = FALSE;    }}/* Return class of symbol SYM.  The returned class can be any of:	0   -> symbol is not interesting to us	'T' -> symbol is a global name	't' -> symbol is a local (static) name.  */static intDEFUN (core_sym_class, (sym), asymbol * sym){  symbol_info syminfo;  const char *name;  char sym_prefix;  int i;  if (sym->section == NULL || (sym->flags & BSF_DEBUGGING) != 0)    return 0;  /* Must be a text symbol, and static text symbols     don't qualify if ignore_static_funcs set.   */  if (ignore_static_funcs && (sym->flags & BSF_LOCAL))    {      DBG (AOUTDEBUG, printf ("[core_sym_class] %s: not a function\n",			      sym->name));      return 0;    }  bfd_get_symbol_info (core_bfd, sym, &syminfo);  i = syminfo.type;  if (i == 'T')    return i;			/* It's a global symbol.  */  if (i == 'W')    /* Treat weak symbols as text symbols.  FIXME: a weak symbol may       also be a data symbol.  */    return 'T';  if (i != 't')    {      /* Not a static text symbol.  */      DBG (AOUTDEBUG, printf ("[core_sym_class] %s is of class %c\n",			      sym->name, i));      return 0;    }  /* Do some more filtering on static function-names.  */  if (ignore_static_funcs)    return 0;  /* Can't zero-length name or funny characters in name, where     `funny' includes: `.' (.o file names) and `$' (Pascal labels).  */  if (!sym->name || sym->name[0] == '\0')    return 0;  for (name = sym->name; *name; ++name)    {      if (*name == '.' || *name == '$')	return 0;    }  /* On systems where the C compiler adds an underscore to all     names, static names without underscores seem usually to be     labels in hand written assembler in the library.  We don't want     these names.  This is certainly necessary on a Sparc running     SunOS 4.1 (try profiling a program that does a lot of     division). I don't know whether it has harmful side effects on     other systems.  Perhaps it should be made configurable.  */  sym_prefix = bfd_get_symbol_leading_char (core_bfd);  if ((sym_prefix && sym_prefix != sym->name[0])      /* GCC may add special symbols to help gdb figure out the file	language.  We want to ignore these, since sometimes they mask	the real function.  (dj@ctron)  */      || !strncmp (sym->name, "__gnu_compiled", 14)      || !strncmp (sym->name, "___gnu_compiled", 15))    {      return 0;    }  /* If the object file supports marking of function symbols, then     we can zap anything that doesn't have BSF_FUNCTION set.  */  if (ignore_non_functions && (sym->flags & BSF_FUNCTION) == 0)    return 0;  return 't';			/* It's a static text symbol.  */}/* Get whatever source info we can get regarding address ADDR.  */static boolDEFUN (get_src_info, (addr, filename, name, line_num),       bfd_vma addr AND const char **filename AND const char **name       AND int *line_num){  const char *fname = 0, *func_name = 0;  int l = 0;  if (bfd_find_nearest_line (core_bfd, core_text_sect, core_syms,			     addr - core_text_sect->vma,			     &fname, &func_name, (unsigned int *) &l)

⌨️ 快捷键说明

复制代码Ctrl + C
搜索代码Ctrl + F
全屏模式F11
增大字号Ctrl + =
减小字号Ctrl + -
显示快捷键?