| /* source.c - Keep track of source files. |
| |
| Copyright (C) 2000-2024 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 3 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., 51 Franklin Street - Fifth Floor, Boston, MA |
| 02110-1301, USA. */ |
| |
| #include "gprof.h" |
| #include "libiberty.h" |
| #include "filenames.h" |
| #include "search_list.h" |
| #include "source.h" |
| |
| #define EXT_ANNO "-ann" /* Postfix of annotated files. */ |
| |
| /* Default option values. */ |
| bool create_annotation_files = false; |
| |
| Search_List src_search_list = {0, 0}; |
| Source_File *first_src_file = 0; |
| |
| |
| Source_File * |
| source_file_lookup_path (const char *path) |
| { |
| Source_File *sf; |
| |
| for (sf = first_src_file; sf; sf = sf->next) |
| { |
| if (FILENAME_CMP (path, sf->name) == 0) |
| break; |
| } |
| |
| if (!sf) |
| { |
| /* Create a new source file descriptor. */ |
| sf = (Source_File *) xmalloc (sizeof (*sf)); |
| |
| memset (sf, 0, sizeof (*sf)); |
| |
| sf->name = xstrdup (path); |
| sf->next = first_src_file; |
| first_src_file = sf; |
| } |
| |
| return sf; |
| } |
| |
| |
| Source_File * |
| source_file_lookup_name (const char *filename) |
| { |
| const char *fname; |
| Source_File *sf; |
| |
| /* The user cannot know exactly how a filename will be stored in |
| the debugging info (e.g., ../include/foo.h |
| vs. /usr/include/foo.h). So we simply compare the filename |
| component of a path only. */ |
| for (sf = first_src_file; sf; sf = sf->next) |
| { |
| fname = strrchr (sf->name, '/'); |
| |
| if (fname) |
| ++fname; |
| else |
| fname = sf->name; |
| |
| if (FILENAME_CMP (filename, fname) == 0) |
| break; |
| } |
| |
| return sf; |
| } |
| |
| |
| FILE * |
| annotate_source (Source_File *sf, unsigned int max_width, |
| void (*annote) (char *, unsigned int, int, void *), |
| void *arg) |
| { |
| static bool first_file = true; |
| int i, line_num, nread; |
| bool new_line; |
| char buf[8192]; |
| char *fname; |
| char *annotation, *name_only; |
| FILE *ifp, *ofp; |
| Search_List_Elem *sle = src_search_list.head; |
| |
| /* Open input file. If open fails, walk along search-list until |
| open succeeds or reaching end of list. */ |
| fname = (char *) sf->name; |
| |
| if (IS_ABSOLUTE_PATH (sf->name)) |
| sle = 0; /* Don't use search list for absolute paths. */ |
| |
| name_only = 0; |
| while (true) |
| { |
| DBG (SRCDEBUG, printf ("[annotate_source]: looking for %s, trying %s\n", |
| sf->name, fname)); |
| |
| ifp = fopen (fname, FOPEN_RB); |
| if (fname != sf->name) |
| free (fname); |
| if (ifp) |
| break; |
| |
| if (!sle && !name_only) |
| { |
| name_only = strrchr (sf->name, '/'); |
| #ifdef HAVE_DOS_BASED_FILE_SYSTEM |
| { |
| char *bslash = strrchr (sf->name, '\\'); |
| if (name_only == NULL || (bslash != NULL && bslash > name_only)) |
| name_only = bslash; |
| if (name_only == NULL && sf->name[0] != '\0' && sf->name[1] == ':') |
| name_only = (char *)sf->name + 1; |
| } |
| #endif |
| if (name_only) |
| { |
| /* Try search-list again, but this time with name only. */ |
| ++name_only; |
| sle = src_search_list.head; |
| } |
| } |
| |
| if (sle) |
| { |
| fname = xmalloc (strlen (sle->path) + 3 |
| + strlen (name_only ? name_only : sf->name)); |
| strcpy (fname, sle->path); |
| #ifdef HAVE_DOS_BASED_FILE_SYSTEM |
| /* d:foo is not the same thing as d:/foo! */ |
| if (fname[strlen (fname) - 1] == ':') |
| strcat (fname, "."); |
| #endif |
| strcat (fname, "/"); |
| |
| if (name_only) |
| strcat (fname, name_only); |
| else |
| strcat (fname, sf->name); |
| |
| sle = sle->next; |
| } |
| else |
| { |
| if (errno == ENOENT) |
| fprintf (stderr, _("%s: could not locate `%s'\n"), |
| whoami, sf->name); |
| else |
| perror (sf->name); |
| |
| return 0; |
| } |
| } |
| |
| ofp = stdout; |
| |
| if (create_annotation_files) |
| { |
| /* Try to create annotated source file. */ |
| const char *filename; |
| |
| /* Create annotation files in the current working directory. */ |
| filename = strrchr (sf->name, '/'); |
| #ifdef HAVE_DOS_BASED_FILE_SYSTEM |
| { |
| char *bslash = strrchr (sf->name, '\\'); |
| if (filename == NULL || (bslash != NULL && bslash > filename)) |
| filename = bslash; |
| if (filename == NULL && sf->name[0] != '\0' && sf->name[1] == ':') |
| filename = sf->name + 1; |
| } |
| #endif |
| if (filename) |
| ++filename; |
| else |
| filename = sf->name; |
| |
| fname = xmalloc (strlen (filename) + strlen (EXT_ANNO) + 1); |
| strcpy (fname, filename); |
| strcat (fname, EXT_ANNO); |
| #ifdef __MSDOS__ |
| { |
| /* foo.cpp-ann can overwrite foo.cpp due to silent truncation of |
| file names on 8+3 filesystems. Their `stat' better be good... */ |
| struct stat buf1, buf2; |
| |
| if (stat (filename, &buf1) == 0 |
| && stat (fname, &buf2) == 0 |
| && buf1.st_ino == buf2.st_ino) |
| { |
| char *dot = strrchr (fname, '.'); |
| |
| if (!dot) |
| dot = fname + strlen (filename); |
| strcpy (dot, ".ann"); |
| } |
| } |
| #endif |
| ofp = fopen (fname, "w"); |
| |
| if (!ofp) |
| { |
| perror (fname); |
| free (fname); |
| return 0; |
| } |
| free (fname); |
| } |
| |
| /* Print file names if output goes to stdout |
| and there are more than one source file. */ |
| if (ofp == stdout) |
| { |
| if (first_file) |
| first_file = false; |
| else |
| fputc ('\n', ofp); |
| |
| if (first_output) |
| first_output = false; |
| else |
| fprintf (ofp, "\f\n"); |
| |
| fprintf (ofp, _("*** File %s:\n"), sf->name); |
| } |
| |
| annotation = (char *) xmalloc (max_width + 1); |
| line_num = 1; |
| new_line = true; |
| |
| while ((nread = fread (buf, 1, sizeof (buf), ifp)) > 0) |
| { |
| for (i = 0; i < nread; ++i) |
| { |
| if (new_line) |
| { |
| (*annote) (annotation, max_width, line_num, arg); |
| fputs (annotation, ofp); |
| ++line_num; |
| } |
| |
| new_line = (buf[i] == '\n'); |
| fputc (buf[i], ofp); |
| } |
| } |
| |
| free (annotation); |
| fclose (ifp); |
| return ofp; |
| } |