mirror of
https://gcc.gnu.org/git/gcc.git
synced 2024-11-23 19:03:59 +08:00
184 lines
5.5 KiB
C++
184 lines
5.5 KiB
C++
/* Implementation of file prefix remapping support (-f*-prefix-map options).
|
|
Copyright (C) 2017-2024 Free Software Foundation, Inc.
|
|
|
|
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, 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; see the file COPYING3. If not see
|
|
<http://www.gnu.org/licenses/>. */
|
|
|
|
#include "config.h"
|
|
#include "system.h"
|
|
#include "coretypes.h"
|
|
#include "diagnostic.h"
|
|
#include "file-prefix-map.h"
|
|
|
|
/* Structure recording the mapping from source file and directory names at
|
|
compile time to those to be embedded in the compilation result (debug
|
|
information, the __FILE__ macro expansion, etc). */
|
|
struct file_prefix_map
|
|
{
|
|
const char *old_prefix;
|
|
const char *new_prefix;
|
|
size_t old_len;
|
|
size_t new_len;
|
|
bool canonicalize;
|
|
struct file_prefix_map *next;
|
|
};
|
|
|
|
/* Record a file prefix mapping in the specified map. ARG is the argument to
|
|
-f*-prefix-map and must be of the form OLD=NEW. OPT is the option name
|
|
for diagnostics. */
|
|
static void
|
|
add_prefix_map (file_prefix_map *&maps, const char *arg, const char *opt)
|
|
{
|
|
file_prefix_map *map;
|
|
const char *p;
|
|
|
|
/* Note: looking for the last '='. The thinking is we can control the paths
|
|
inside our projects but not where the users build them. */
|
|
p = strrchr (arg, '=');
|
|
if (!p)
|
|
{
|
|
error ("invalid argument %qs to %qs", arg, opt);
|
|
return;
|
|
}
|
|
map = XNEW (file_prefix_map);
|
|
map->canonicalize = flag_canon_prefix_map;
|
|
map->old_prefix = xstrndup (arg, p - arg);
|
|
map->old_len = p - arg;
|
|
if (map->canonicalize)
|
|
{
|
|
char *realname = lrealpath (map->old_prefix);
|
|
free (const_cast <char *> (map->old_prefix));
|
|
map->old_prefix = realname;
|
|
map->old_len = strlen (realname);
|
|
}
|
|
p++;
|
|
map->new_prefix = xstrdup (p);
|
|
map->new_len = strlen (p);
|
|
map->next = maps;
|
|
maps = map;
|
|
}
|
|
|
|
/* Perform user-specified mapping of filename prefixes. Return the
|
|
GC-allocated new name corresponding to FILENAME or FILENAME if no
|
|
remapping was performed. */
|
|
|
|
static const char *
|
|
remap_filename (file_prefix_map *maps, const char *filename)
|
|
{
|
|
file_prefix_map *map;
|
|
char *s;
|
|
const char *name;
|
|
const char *realname = NULL;
|
|
size_t name_len;
|
|
|
|
if (!filename)
|
|
return filename;
|
|
|
|
for (map = maps; map; map = map->next)
|
|
if (map->canonicalize)
|
|
{
|
|
if (realname == NULL)
|
|
{
|
|
if (lbasename (filename) == filename)
|
|
realname = filename;
|
|
else
|
|
realname = lrealpath (filename);
|
|
}
|
|
if (filename_ncmp (realname, map->old_prefix, map->old_len) == 0)
|
|
break;
|
|
}
|
|
else if (filename_ncmp (filename, map->old_prefix, map->old_len) == 0)
|
|
break;
|
|
if (!map)
|
|
{
|
|
if (realname != filename)
|
|
free (const_cast <char *> (realname));
|
|
return filename;
|
|
}
|
|
if (map->canonicalize)
|
|
name = realname + map->old_len;
|
|
else
|
|
name = filename + map->old_len;
|
|
name_len = strlen (name) + 1;
|
|
|
|
s = (char *) ggc_alloc_atomic (name_len + map->new_len);
|
|
memcpy (s, map->new_prefix, map->new_len);
|
|
memcpy (s + map->new_len, name, name_len);
|
|
if (realname != filename)
|
|
free (const_cast <char *> (realname));
|
|
return s;
|
|
}
|
|
|
|
/* NOTE: if adding another -f*-prefix-map option then don't forget to
|
|
ignore it in DW_AT_producer (gen_command_line_string in opts.cc). */
|
|
|
|
/* Linked lists of file_prefix_map structures. */
|
|
static file_prefix_map *macro_prefix_maps; /* -fmacro-prefix-map */
|
|
static file_prefix_map *debug_prefix_maps; /* -fdebug-prefix-map */
|
|
static file_prefix_map *profile_prefix_maps; /* -fprofile-prefix-map */
|
|
|
|
/* Record a file prefix mapping for -fmacro-prefix-map. */
|
|
void
|
|
add_macro_prefix_map (const char *arg)
|
|
{
|
|
add_prefix_map (macro_prefix_maps, arg, "-fmacro-prefix-map");
|
|
}
|
|
|
|
/* Record a file prefix mapping for -fdebug-prefix-map. */
|
|
void
|
|
add_debug_prefix_map (const char *arg)
|
|
{
|
|
add_prefix_map (debug_prefix_maps, arg, "-fdebug-prefix-map");
|
|
}
|
|
|
|
/* Record a file prefix mapping for all -f*-prefix-map. */
|
|
void
|
|
add_file_prefix_map (const char *arg)
|
|
{
|
|
add_prefix_map (macro_prefix_maps, arg, "-ffile-prefix-map");
|
|
add_prefix_map (debug_prefix_maps, arg, "-ffile-prefix-map");
|
|
add_prefix_map (profile_prefix_maps, arg, "-ffile-prefix-map");
|
|
}
|
|
|
|
/* Record a file prefix mapping for -fprofile-prefix-map. */
|
|
void
|
|
add_profile_prefix_map (const char *arg)
|
|
{
|
|
add_prefix_map (profile_prefix_maps, arg, "-fprofile-prefix-map");
|
|
}
|
|
|
|
/* Remap using -fmacro-prefix-map. Return the GC-allocated new name
|
|
corresponding to FILENAME or FILENAME if no remapping was performed. */
|
|
const char *
|
|
remap_macro_filename (const char *filename)
|
|
{
|
|
return remap_filename (macro_prefix_maps, filename);
|
|
}
|
|
|
|
/* Remap using -fdebug-prefix-map. Return the GC-allocated new name
|
|
corresponding to FILENAME or FILENAME if no remapping was performed. */
|
|
const char *
|
|
remap_debug_filename (const char *filename)
|
|
{
|
|
return remap_filename (debug_prefix_maps, filename);
|
|
}
|
|
|
|
/* Remap using -fprofile-prefix-map. Return the GC-allocated new name
|
|
corresponding to FILENAME or FILENAME if no remapping was performed. */
|
|
const char *
|
|
remap_profile_filename (const char *filename)
|
|
{
|
|
return remap_filename (profile_prefix_maps, filename);
|
|
}
|