Regenerate configure to include the gettext.m4 change. Update
[platform/upstream/binutils.git] / gprof / gprof.c
index 1e3226f..4a1fd50 100644 (file)
@@ -1,5 +1,5 @@
 /*
- * Copyright (c) 1983 Regents of the University of California.
+ * Copyright (c) 1983, 1998, 2001 Regents of the University of California.
  * All rights reserved.
  *
  * Redistribution and use in source and binary forms are permitted
 #include "call_graph.h"
 #include "cg_arcs.h"
 #include "cg_print.h"
-#include "core.h"
+#include "corefile.h"
 #include "gmon_io.h"
 #include "hertz.h"
 #include "hist.h"
 #include "source.h"
 #include "sym_ids.h"
-
-#define VERSION "2.6"
+#include "demangle.h"
 
 const char *whoami;
+const char *function_mapping_file;
 const char *a_out_name = A_OUTNAME;
 long hz = HZ_WRONG;
 
@@ -43,6 +43,7 @@ int debug_level = 0;
 int output_style = 0;
 int output_width = 80;
 bool bsd_style_output = FALSE;
+bool demangle = TRUE;
 bool discard_underscores = TRUE;
 bool ignore_direct_calls = FALSE;
 bool ignore_static_funcs = FALSE;
@@ -50,12 +51,13 @@ bool ignore_zeros = TRUE;
 bool line_granularity = FALSE;
 bool print_descriptions = TRUE;
 bool print_path = FALSE;
+bool ignore_non_functions = FALSE;
 File_Format file_format = FF_AUTO;
 
 bool first_output = TRUE;
 
 char copyright[] =
-"@(#) Copyright (c) 1983 Regents of the University of California.\n\
+ "@(#) Copyright (c) 1983 Regents of the University of California.\n\
  All rights reserved.\n";
 
 static char *gmon_name = GMONNAME;     /* profile filename */
@@ -67,15 +69,23 @@ bfd *abfd;
  */
 static char *default_excluded_list[] =
 {
-  "_gprof_mcount", "mcount", "_mcount", "__mcleanup",
+  "_gprof_mcount", "mcount", "_mcount", "__mcount", "__mcount_internal",
+  "__mcleanup",
   "<locore>", "<hicore>",
   0
 };
 
+/* Codes used for the long options with no short synonyms.  150 isn't
+   special; it's just an arbitrary non-ASCII char value.  */
+
+#define OPTION_DEMANGLE                (150)
+#define OPTION_NO_DEMANGLE     (OPTION_DEMANGLE + 1)
+
 static struct option long_options[] =
 {
   {"line", no_argument, 0, 'l'},
   {"no-static", no_argument, 0, 'a'},
+  {"ignore-non-functions", no_argument, 0, 'D'},
 
     /* output styles: */
 
@@ -87,12 +97,16 @@ static struct option long_options[] =
   {"no-graph", optional_argument, 0, 'Q'},
   {"exec-counts", optional_argument, 0, 'C'},
   {"no-exec-counts", optional_argument, 0, 'Z'},
+  {"function-ordering", no_argument, 0, 'r'},
+  {"file-ordering", required_argument, 0, 'R'},
   {"file-info", no_argument, 0, 'i'},
   {"sum", no_argument, 0, 's'},
 
     /* various options to affect output: */
 
   {"all-lines", no_argument, 0, 'x'},
+  {"demangle", optional_argument, 0, OPTION_DEMANGLE},
+  {"no-demangle", no_argument, 0, OPTION_NO_DEMANGLE},
   {"directory-path", required_argument, 0, 'I'},
   {"display-unused-functions", no_argument, 0, 'z'},
   {"min-count", required_argument, 0, 'm'},
@@ -128,21 +142,25 @@ static struct option long_options[] =
 static void
 DEFUN (usage, (stream, status), FILE * stream AND int status)
 {
-  fprintf (stream, "\
-Usage: %s [-[abchilLsTvwxyz]] [-[ACeEfFJnNOpPqQZ][name]] [-I dirs]\n\
+  fprintf (stream, _("\
+Usage: %s [-[abcDhilLsTvwxyz]] [-[ACeEfFJnNOpPqQZ][name]] [-I dirs]\n\
        [-d[num]] [-k from/to] [-m min-count] [-t table-length]\n\
        [--[no-]annotated-source[=name]] [--[no-]exec-counts[=name]]\n\
        [--[no-]flat-profile[=name]] [--[no-]graph[=name]]\n\
        [--[no-]time=name] [--all-lines] [--brief] [--debug[=level]]\n\
+       [--function-ordering] [--file-ordering]\n\
        [--directory-path=dirs] [--display-unused-functions]\n\
        [--file-format=name] [--file-info] [--help] [--line] [--min-count=n]\n\
        [--no-static] [--print-path] [--separate-files]\n\
        [--static-call-graph] [--sum] [--table-length=len] [--traditional]\n\
-       [--version] [--width=n]\n\
-       [image-file] [profile-file...]\n",
+       [--version] [--width=n] [--ignore-non-functions]\n\
+       [--demangle[=STYLE]] [--no-demangle]\n\
+       [image-file] [profile-file...]\n"),
           whoami);
+  if (status == 0)
+    fprintf (stream, _("Report bugs to %s\n"), REPORT_BUGS_TO);
   done (status);
-}                              /* usage */
+}
 
 
 int
@@ -152,11 +170,20 @@ DEFUN (main, (argc, argv), int argc AND char **argv)
   Sym **cg = 0;
   int ch, user_specified = 0;
 
+#if defined (HAVE_SETLOCALE) && defined (HAVE_LC_MESSAGES)
+  setlocale (LC_MESSAGES, "");
+#endif
+#if defined (HAVE_SETLOCALE)
+  setlocale (LC_CTYPE, "");
+#endif
+  bindtextdomain (PACKAGE, LOCALEDIR);
+  textdomain (PACKAGE);
+
   whoami = argv[0];
   xmalloc_set_program_name (whoami);
 
   while ((ch = getopt_long (argc, argv,
-       "aA::bBcCd::e:E:f:F:hiI:J::k:lLm:n::N::O:p::P::q::Q::st:Tvw:xyzZ::",
+       "aA::bBcCd::De:E:f:F:hiI:J::k:lLm:n::N::O:p::P::q::Q::st:Tvw:xyzZ::",
                            long_options, 0))
         != EOF)
     {
@@ -169,7 +196,7 @@ DEFUN (main, (argc, argv), int argc AND char **argv)
          if (optarg)
            {
              sym_id_add (optarg, INCL_ANNO);
-           }                   /* if */
+           }
          output_style |= STYLE_ANNOTATED_SOURCE;
          user_specified |= STYLE_ANNOTATED_SOURCE;
          break;
@@ -187,7 +214,7 @@ DEFUN (main, (argc, argv), int argc AND char **argv)
          if (optarg)
            {
              sym_id_add (optarg, INCL_EXEC);
-           }                   /* if */
+           }
          output_style |= STYLE_EXEC_COUNTS;
          user_specified |= STYLE_EXEC_COUNTS;
          break;
@@ -200,12 +227,15 @@ DEFUN (main, (argc, argv), int argc AND char **argv)
          else
            {
              debug_level = ~0;
-           }                   /* if */
+           }
          DBG (ANYDEBUG, printf ("[main] debug-level=0x%x\n", debug_level));
 #ifndef DEBUG
-         printf ("%s: debugging not supported; -d ignored\n", whoami);
+         printf (_("%s: debugging not supported; -d ignored\n"), whoami);
 #endif /* DEBUG */
          break;
+       case 'D':
+         ignore_non_functions = TRUE;
+         break;
        case 'E':
          sym_id_add (optarg, EXCL_TIME);
        case 'e':
@@ -240,7 +270,7 @@ DEFUN (main, (argc, argv), int argc AND char **argv)
          else
            {
              output_style &= ~STYLE_ANNOTATED_SOURCE;
-           }                   /* if */
+           }
          user_specified |= STYLE_ANNOTATED_SOURCE;
          break;
        case 'k':
@@ -253,7 +283,7 @@ DEFUN (main, (argc, argv), int argc AND char **argv)
          print_path = TRUE;
          break;
        case 'm':
-         bb_min_calls = atoi (optarg);
+         bb_min_calls = (unsigned long) strtoul (optarg, (char **) NULL, 10);
          break;
        case 'n':
          sym_id_add (optarg, INCL_TIME);
@@ -273,20 +303,23 @@ DEFUN (main, (argc, argv), int argc AND char **argv)
            case 'b':
              file_format = FF_BSD;
              break;
+           case '4':
+             file_format = FF_BSD44;
+             break;
            case 'p':
              file_format = FF_PROF;
              break;
            default:
-             fprintf (stderr, "%s: unknown file format %s\n",
+             fprintf (stderr, _("%s: unknown file format %s\n"),
                       optarg, whoami);
              done (1);
-           }                   /* switch */
+           }
          break;
        case 'p':
          if (optarg)
            {
              sym_id_add (optarg, INCL_FLAT);
-           }                   /* if */
+           }
          output_style |= STYLE_FLAT_PROFILE;
          user_specified |= STYLE_FLAT_PROFILE;
          break;
@@ -299,7 +332,7 @@ DEFUN (main, (argc, argv), int argc AND char **argv)
          else
            {
              output_style &= ~STYLE_FLAT_PROFILE;
-           }                   /* if */
+           }
          user_specified |= STYLE_FLAT_PROFILE;
          break;
        case 'q':
@@ -312,11 +345,20 @@ DEFUN (main, (argc, argv), int argc AND char **argv)
              else
                {
                  sym_id_add (optarg, INCL_GRAPH);
-               }               /* if */
-           }                   /* if */
+               }
+           }
          output_style |= STYLE_CALL_GRAPH;
          user_specified |= STYLE_CALL_GRAPH;
          break;
+       case 'r':
+         output_style |= STYLE_FUNCTION_ORDER;
+         user_specified |= STYLE_FUNCTION_ORDER;
+         break;
+       case 'R':
+         output_style |= STYLE_FILE_ORDER;
+         user_specified |= STYLE_FILE_ORDER;
+         function_mapping_file = optarg;
+         break;
        case 'Q':
          if (optarg)
            {
@@ -327,13 +369,13 @@ DEFUN (main, (argc, argv), int argc AND char **argv)
              else
                {
                  sym_id_add (optarg, EXCL_GRAPH);
-               }               /* if */
+               }
              output_style |= STYLE_CALL_GRAPH;
            }
          else
            {
              output_style &= ~STYLE_CALL_GRAPH;
-           }                   /* if */
+           }
          user_specified |= STYLE_CALL_GRAPH;
          break;
        case 's':
@@ -345,20 +387,24 @@ DEFUN (main, (argc, argv), int argc AND char **argv)
          if (bb_table_length < 0)
            {
              bb_table_length = 0;
-           }                   /* if */
+           }
          break;
        case 'T':
          bsd_style_output = TRUE;
          break;
        case 'v':
-         printf ("%s version %s\n", whoami, VERSION);
+         /* This output is intended to follow the GNU standards document.  */
+         printf (_("GNU gprof %s\n"), VERSION);
+         printf (_("Based on BSD gprof, copyright 1983 Regents of the University of California.\n"));
+         printf (_("\
+This program is free software.  This program has absolutely no warranty.\n"));
          done (0);
        case 'w':
          output_width = atoi (optarg);
          if (output_width < 1)
            {
              output_width = 1;
-           }                   /* if */
+           }
          break;
        case 'x':
          bb_annotate_all_lines = TRUE;
@@ -378,29 +424,66 @@ DEFUN (main, (argc, argv), int argc AND char **argv)
          else
            {
              output_style &= ~STYLE_EXEC_COUNTS;
-           }                   /* if */
+           }
          user_specified |= STYLE_ANNOTATED_SOURCE;
          break;
+       case OPTION_DEMANGLE:
+         demangle = TRUE;
+         if (optarg != NULL)
+           {
+             enum demangling_styles style;
+
+             style = cplus_demangle_name_to_style (optarg);
+             if (style == unknown_demangling)
+               {
+                 fprintf (stderr,
+                          _("%s: unknown demangling style `%s'\n"),
+                          whoami, optarg);
+                 xexit (1);
+               }
+
+             cplus_demangle_set_style (style);
+          }
+         break;
+       case OPTION_NO_DEMANGLE:
+         demangle = FALSE;
+         break;
        default:
          usage (stderr, 1);
-       }                       /* switch */
-    }                          /* while */
+       }
+    }
+
+  /* Don't allow both ordering options, they modify the arc data in-place.  */
+  if ((user_specified & STYLE_FUNCTION_ORDER)
+      && (user_specified & STYLE_FILE_ORDER))
+    {
+      fprintf (stderr,_("\
+%s: Only one of --function-ordering and --file-ordering may be specified.\n"),
+              whoami);
+      done (1);
+    }
+
+  /* --sum implies --line, otherwise we'd lose b-b counts in gmon.sum */
+  if (output_style & STYLE_SUMMARY_FILE)
+    {
+      line_granularity = 1;
+    }
 
   /* append value of GPROF_PATH to source search list if set: */
-  str = getenv ("GPROF_PATH");
+  str = (char *) getenv ("GPROF_PATH");
   if (str)
     {
       search_list_append (&src_search_list, str);
-    }                          /* if */
+    }
 
   if (optind < argc)
     {
       a_out_name = argv[optind++];
-    }                          /* if */
+    }
   if (optind < argc)
     {
       gmon_name = argv[optind++];
-    }                          /* if */
+    }
 
   /*
    * Turn off default functions:
@@ -409,10 +492,10 @@ DEFUN (main, (argc, argv), int argc AND char **argv)
     {
       sym_id_add (*sp, EXCL_TIME);
       sym_id_add (*sp, EXCL_GRAPH);
-#ifdef __osf__
+#ifdef __alpha__
       sym_id_add (*sp, EXCL_FLAT);
 #endif
-    }                          /* for */
+    }
 
   /*
    * For line-by-line profiling, also want to keep those
@@ -423,8 +506,8 @@ DEFUN (main, (argc, argv), int argc AND char **argv)
       for (sp = &default_excluded_list[0]; *sp; sp++)
        {
          sym_id_add (*sp, EXCL_FLAT);
-       }                       /* for */
-    }                          /* if */
+       }
+    }
 
   /*
    * Read symbol table from core file:
@@ -438,7 +521,7 @@ DEFUN (main, (argc, argv), int argc AND char **argv)
   if (ignore_direct_calls)
     {
       core_get_text_space (core_bfd);
-    }                          /* if */
+    }
 
   /*
    * Create symbols from core image:
@@ -450,7 +533,7 @@ DEFUN (main, (argc, argv), int argc AND char **argv)
   else
     {
       core_create_function_syms (core_bfd);
-    }                          /* if */
+    }
 
   /*
    * Translate sym specs into syms:
@@ -469,12 +552,12 @@ DEFUN (main, (argc, argv), int argc AND char **argv)
          if (optind < argc)
            {
              gmon_name = argv[optind];
-           }                   /* if */
+           }
        }
       while (optind++ < argc);
 #else
       fprintf (stderr,
-              "%s: sorry, file format `prof' is not yet supported\n",
+              _("%s: sorry, file format `prof' is not yet supported\n"),
               whoami);
       done (1);
 #endif
@@ -490,10 +573,10 @@ DEFUN (main, (argc, argv), int argc AND char **argv)
          if (optind < argc)
            {
              gmon_name = argv[optind];
-           }                   /* if */
+           }
        }
       while (optind++ < argc);
-    }                          /* if */
+    }
 
   /*
    * If user did not specify output style, try to guess something
@@ -508,9 +591,9 @@ DEFUN (main, (argc, argv), int argc AND char **argv)
       else
        {
          output_style = STYLE_EXEC_COUNTS;
-       }                       /* if */
+       }
       output_style &= ~user_specified;
-    }                          /* if */
+    }
 
   /*
    * Dump a gmon.sum file if requested (before any other processing!):
@@ -518,64 +601,72 @@ DEFUN (main, (argc, argv), int argc AND char **argv)
   if (output_style & STYLE_SUMMARY_FILE)
     {
       gmon_out_write (GMONSUM);
-    }                          /* if */
+    }
 
   if (gmon_input & INPUT_HISTOGRAM)
     {
       hist_assign_samples ();
-    }                          /* if */
+    }
 
   if (gmon_input & INPUT_CALL_GRAPH)
     {
       cg = cg_assemble ();
-    }                          /* if */
+    }
 
   /* do some simple sanity checks: */
 
   if ((output_style & STYLE_FLAT_PROFILE)
       && !(gmon_input & INPUT_HISTOGRAM))
     {
-      fprintf (stderr, "%s: gmon.out file is missing histogram\n", whoami);
+      fprintf (stderr, _("%s: gmon.out file is missing histogram\n"), whoami);
       done (1);
-    }                          /* if */
+    }
 
   if ((output_style & STYLE_CALL_GRAPH) && !(gmon_input & INPUT_CALL_GRAPH))
     {
       fprintf (stderr,
-              "%s: gmon.out file is missing call-graph data\n", whoami);
+              _("%s: gmon.out file is missing call-graph data\n"), whoami);
       done (1);
-    }                          /* if */
+    }
 
   /* output whatever user whishes to see: */
 
   if (cg && (output_style & STYLE_CALL_GRAPH) && bsd_style_output)
     {
       cg_print (cg);           /* print the dynamic profile */
-    }                          /* if */
+    }
 
   if (output_style & STYLE_FLAT_PROFILE)
     {
       hist_print ();           /* print the flat profile */
-    }                          /* if */
+    }
 
   if (cg && (output_style & STYLE_CALL_GRAPH))
     {
       if (!bsd_style_output)
        {
          cg_print (cg);        /* print the dynamic profile */
-       }                       /* if */
+       }
       cg_print_index ();
-    }                          /* if */
+    }
 
   if (output_style & STYLE_EXEC_COUNTS)
     {
       print_exec_counts ();
-    }                          /* if */
+    }
 
   if (output_style & STYLE_ANNOTATED_SOURCE)
     {
       print_annotated_source ();
-    }                          /* if */
+    }
+  if (output_style & STYLE_FUNCTION_ORDER)
+    {
+      cg_print_function_ordering ();
+    }
+  if (output_style & STYLE_FILE_ORDER)
+    {
+      cg_print_file_ordering ();
+    }
   return 0;
 }