mirror of
https://sourceware.org/git/binutils-gdb.git
synced 2025-01-06 12:09:26 +08:00
6a7f1c20e8
We currently have scoped_restore_sigttou and scoped_restore_sigpipe doing basically the same thing -- temporarily ignoring a specific signal. This patch introduce a scoped_restore_signal type that can be used for both. This will become more important for the next patch which changes how the signal-ignoring is implemented. scoped_restore_sigpipe is a straight alias to scoped_restore_signal<SIGPIPE> on systems that define SIGPIPE, and an alias to scoped_restore_signal_nop (a no-op version of scoped_restore_signal) otherwise. scoped_restore_sigttou is not a straight alias because it wants to check the job_control global. gdb/ChangeLog: yyyy-mm-dd Pedro Alves <pedro@palves.net> * gdbsupport/scoped_ignore_signal.h: New. * compile/compile.c: Include gdbsupport/scoped_ignore_signal.h instead of <signal.h>. Don't include <unistd.h>. (scoped_ignore_sigpipe): Remove. * gdbsupport/scoped_ignore_sigttou.h: Include gdbsupport/scoped_ignore_signal.h instead of <signal.h>. Don't include <unistd.h>. (lazy_init): New. (scoped_ignore_sigttou): Reimplement using scoped_ignore_signal and lazy_init. Change-Id: Ibb44d0bd705e96df03ef0787c77358a4a7b7086c
1063 lines
29 KiB
C
1063 lines
29 KiB
C
/* General Compile and inject code
|
||
|
||
Copyright (C) 2014-2021 Free Software Foundation, Inc.
|
||
|
||
This file is part of GDB.
|
||
|
||
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, see <http://www.gnu.org/licenses/>. */
|
||
|
||
#include "defs.h"
|
||
#include "top.h"
|
||
#include "ui-out.h"
|
||
#include "command.h"
|
||
#include "cli/cli-script.h"
|
||
#include "cli/cli-utils.h"
|
||
#include "cli/cli-option.h"
|
||
#include "completer.h"
|
||
#include "gdbcmd.h"
|
||
#include "compile.h"
|
||
#include "compile-internal.h"
|
||
#include "compile-object-load.h"
|
||
#include "compile-object-run.h"
|
||
#include "language.h"
|
||
#include "frame.h"
|
||
#include "source.h"
|
||
#include "block.h"
|
||
#include "arch-utils.h"
|
||
#include "gdbsupport/filestuff.h"
|
||
#include "target.h"
|
||
#include "osabi.h"
|
||
#include "gdbsupport/gdb_wait.h"
|
||
#include "valprint.h"
|
||
#include "gdbsupport/gdb_optional.h"
|
||
#include "gdbsupport/gdb_unlinker.h"
|
||
#include "gdbsupport/pathstuff.h"
|
||
#include "gdbsupport/scoped_ignore_signal.h"
|
||
|
||
|
||
|
||
/* Initial filename for temporary files. */
|
||
|
||
#define TMP_PREFIX "/tmp/gdbobj-"
|
||
|
||
/* Hold "compile" commands. */
|
||
|
||
static struct cmd_list_element *compile_command_list;
|
||
|
||
/* Debug flag for "compile" commands. */
|
||
|
||
bool compile_debug;
|
||
|
||
/* Object of this type are stored in the compiler's symbol_err_map. */
|
||
|
||
struct symbol_error
|
||
{
|
||
/* The symbol. */
|
||
|
||
const struct symbol *sym;
|
||
|
||
/* The error message to emit. This is malloc'd and owned by the
|
||
hash table. */
|
||
|
||
char *message;
|
||
};
|
||
|
||
/* Hash a type_map_instance. */
|
||
|
||
static hashval_t
|
||
hash_type_map_instance (const void *p)
|
||
{
|
||
const struct type_map_instance *inst = (const struct type_map_instance *) p;
|
||
|
||
return htab_hash_pointer (inst->type);
|
||
}
|
||
|
||
/* Check two type_map_instance objects for equality. */
|
||
|
||
static int
|
||
eq_type_map_instance (const void *a, const void *b)
|
||
{
|
||
const struct type_map_instance *insta = (const struct type_map_instance *) a;
|
||
const struct type_map_instance *instb = (const struct type_map_instance *) b;
|
||
|
||
return insta->type == instb->type;
|
||
}
|
||
|
||
/* Hash function for struct symbol_error. */
|
||
|
||
static hashval_t
|
||
hash_symbol_error (const void *a)
|
||
{
|
||
const struct symbol_error *se = (const struct symbol_error *) a;
|
||
|
||
return htab_hash_pointer (se->sym);
|
||
}
|
||
|
||
/* Equality function for struct symbol_error. */
|
||
|
||
static int
|
||
eq_symbol_error (const void *a, const void *b)
|
||
{
|
||
const struct symbol_error *sea = (const struct symbol_error *) a;
|
||
const struct symbol_error *seb = (const struct symbol_error *) b;
|
||
|
||
return sea->sym == seb->sym;
|
||
}
|
||
|
||
/* Deletion function for struct symbol_error. */
|
||
|
||
static void
|
||
del_symbol_error (void *a)
|
||
{
|
||
struct symbol_error *se = (struct symbol_error *) a;
|
||
|
||
xfree (se->message);
|
||
xfree (se);
|
||
}
|
||
|
||
/* Constructor for compile_instance. */
|
||
|
||
compile_instance::compile_instance (struct gcc_base_context *gcc_fe,
|
||
const char *options)
|
||
: m_gcc_fe (gcc_fe), m_gcc_target_options (options),
|
||
m_type_map (htab_create_alloc (10, hash_type_map_instance,
|
||
eq_type_map_instance,
|
||
xfree, xcalloc, xfree)),
|
||
m_symbol_err_map (htab_create_alloc (10, hash_symbol_error,
|
||
eq_symbol_error, del_symbol_error,
|
||
xcalloc, xfree))
|
||
{
|
||
}
|
||
|
||
/* See compile-internal.h. */
|
||
|
||
bool
|
||
compile_instance::get_cached_type (struct type *type, gcc_type *ret) const
|
||
{
|
||
struct type_map_instance inst, *found;
|
||
|
||
inst.type = type;
|
||
found = (struct type_map_instance *) htab_find (m_type_map.get (), &inst);
|
||
if (found != NULL)
|
||
{
|
||
*ret = found->gcc_type_handle;
|
||
return true;
|
||
}
|
||
|
||
return false;
|
||
}
|
||
|
||
/* See compile-internal.h. */
|
||
|
||
void
|
||
compile_instance::insert_type (struct type *type, gcc_type gcc_type)
|
||
{
|
||
struct type_map_instance inst, *add;
|
||
void **slot;
|
||
|
||
inst.type = type;
|
||
inst.gcc_type_handle = gcc_type;
|
||
slot = htab_find_slot (m_type_map.get (), &inst, INSERT);
|
||
|
||
add = (struct type_map_instance *) *slot;
|
||
/* The type might have already been inserted in order to handle
|
||
recursive types. */
|
||
if (add != NULL && add->gcc_type_handle != gcc_type)
|
||
error (_("Unexpected type id from GCC, check you use recent enough GCC."));
|
||
|
||
if (add == NULL)
|
||
{
|
||
add = XNEW (struct type_map_instance);
|
||
*add = inst;
|
||
*slot = add;
|
||
}
|
||
}
|
||
|
||
/* See compile-internal.h. */
|
||
|
||
void
|
||
compile_instance::insert_symbol_error (const struct symbol *sym,
|
||
const char *text)
|
||
{
|
||
struct symbol_error e;
|
||
void **slot;
|
||
|
||
e.sym = sym;
|
||
slot = htab_find_slot (m_symbol_err_map.get (), &e, INSERT);
|
||
if (*slot == NULL)
|
||
{
|
||
struct symbol_error *ep = XNEW (struct symbol_error);
|
||
|
||
ep->sym = sym;
|
||
ep->message = xstrdup (text);
|
||
*slot = ep;
|
||
}
|
||
}
|
||
|
||
/* See compile-internal.h. */
|
||
|
||
void
|
||
compile_instance::error_symbol_once (const struct symbol *sym)
|
||
{
|
||
struct symbol_error search;
|
||
struct symbol_error *err;
|
||
|
||
if (m_symbol_err_map == NULL)
|
||
return;
|
||
|
||
search.sym = sym;
|
||
err = (struct symbol_error *) htab_find (m_symbol_err_map.get (), &search);
|
||
if (err == NULL || err->message == NULL)
|
||
return;
|
||
|
||
gdb::unique_xmalloc_ptr<char> message (err->message);
|
||
err->message = NULL;
|
||
error (_("%s"), message.get ());
|
||
}
|
||
|
||
/* Implement "show debug compile". */
|
||
|
||
static void
|
||
show_compile_debug (struct ui_file *file, int from_tty,
|
||
struct cmd_list_element *c, const char *value)
|
||
{
|
||
fprintf_filtered (file, _("Compile debugging is %s.\n"), value);
|
||
}
|
||
|
||
|
||
|
||
/* Options for the compile command. */
|
||
|
||
struct compile_options
|
||
{
|
||
/* For -raw. */
|
||
bool raw = false;
|
||
};
|
||
|
||
using compile_flag_option_def
|
||
= gdb::option::flag_option_def<compile_options>;
|
||
|
||
static const gdb::option::option_def compile_command_option_defs[] = {
|
||
|
||
compile_flag_option_def {
|
||
"raw",
|
||
[] (compile_options *opts) { return &opts->raw; },
|
||
N_("Suppress automatic 'void _gdb_expr () { CODE }' wrapping."),
|
||
},
|
||
|
||
};
|
||
|
||
/* Create an option_def_group for the "compile" command's options,
|
||
with OPTS as context. */
|
||
|
||
static gdb::option::option_def_group
|
||
make_compile_options_def_group (compile_options *opts)
|
||
{
|
||
return {{compile_command_option_defs}, opts};
|
||
}
|
||
|
||
/* Handle the input from the 'compile file' command. The "compile
|
||
file" command is used to evaluate an expression contained in a file
|
||
that may contain calls to the GCC compiler. */
|
||
|
||
static void
|
||
compile_file_command (const char *args, int from_tty)
|
||
{
|
||
scoped_restore save_async = make_scoped_restore (¤t_ui->async, 0);
|
||
|
||
/* Check if a -raw option is provided. */
|
||
|
||
compile_options options;
|
||
|
||
const gdb::option::option_def_group group
|
||
= make_compile_options_def_group (&options);
|
||
gdb::option::process_options
|
||
(&args, gdb::option::PROCESS_OPTIONS_UNKNOWN_IS_ERROR,
|
||
group);
|
||
|
||
enum compile_i_scope_types scope
|
||
= options.raw ? COMPILE_I_RAW_SCOPE : COMPILE_I_SIMPLE_SCOPE;
|
||
|
||
args = skip_spaces (args);
|
||
|
||
/* After processing options, check whether we have a filename. */
|
||
if (args == nullptr || args[0] == '\0')
|
||
error (_("You must provide a filename for this command."));
|
||
|
||
args = skip_spaces (args);
|
||
gdb::unique_xmalloc_ptr<char> abspath = gdb_abspath (args);
|
||
std::string buffer = string_printf ("#include \"%s\"\n", abspath.get ());
|
||
eval_compile_command (NULL, buffer.c_str (), scope, NULL);
|
||
}
|
||
|
||
/* Completer for the "compile file" command. */
|
||
|
||
static void
|
||
compile_file_command_completer (struct cmd_list_element *ignore,
|
||
completion_tracker &tracker,
|
||
const char *text, const char *word)
|
||
{
|
||
const gdb::option::option_def_group group
|
||
= make_compile_options_def_group (nullptr);
|
||
if (gdb::option::complete_options
|
||
(tracker, &text, gdb::option::PROCESS_OPTIONS_UNKNOWN_IS_ERROR, group))
|
||
return;
|
||
|
||
word = advance_to_filename_complete_word_point (tracker, text);
|
||
filename_completer (ignore, tracker, text, word);
|
||
}
|
||
|
||
/* Handle the input from the 'compile code' command. The
|
||
"compile code" command is used to evaluate an expression that may
|
||
contain calls to the GCC compiler. The language expected in this
|
||
compile command is the language currently set in GDB. */
|
||
|
||
static void
|
||
compile_code_command (const char *args, int from_tty)
|
||
{
|
||
scoped_restore save_async = make_scoped_restore (¤t_ui->async, 0);
|
||
|
||
compile_options options;
|
||
|
||
const gdb::option::option_def_group group
|
||
= make_compile_options_def_group (&options);
|
||
gdb::option::process_options
|
||
(&args, gdb::option::PROCESS_OPTIONS_UNKNOWN_IS_ERROR, group);
|
||
|
||
enum compile_i_scope_types scope
|
||
= options.raw ? COMPILE_I_RAW_SCOPE : COMPILE_I_SIMPLE_SCOPE;
|
||
|
||
if (args && *args)
|
||
eval_compile_command (NULL, args, scope, NULL);
|
||
else
|
||
{
|
||
counted_command_line l = get_command_line (compile_control, "");
|
||
|
||
l->control_u.compile.scope = scope;
|
||
execute_control_command_untraced (l.get ());
|
||
}
|
||
}
|
||
|
||
/* Completer for the "compile code" command. */
|
||
|
||
static void
|
||
compile_code_command_completer (struct cmd_list_element *ignore,
|
||
completion_tracker &tracker,
|
||
const char *text, const char *word)
|
||
{
|
||
const gdb::option::option_def_group group
|
||
= make_compile_options_def_group (nullptr);
|
||
if (gdb::option::complete_options
|
||
(tracker, &text, gdb::option::PROCESS_OPTIONS_UNKNOWN_IS_ERROR, group))
|
||
return;
|
||
|
||
word = advance_to_expression_complete_word_point (tracker, text);
|
||
symbol_completer (ignore, tracker, text, word);
|
||
}
|
||
|
||
/* Callback for compile_print_command. */
|
||
|
||
void
|
||
compile_print_value (struct value *val, void *data_voidp)
|
||
{
|
||
const value_print_options *print_opts = (value_print_options *) data_voidp;
|
||
|
||
print_value (val, *print_opts);
|
||
}
|
||
|
||
/* Handle the input from the 'compile print' command. The "compile
|
||
print" command is used to evaluate and print an expression that may
|
||
contain calls to the GCC compiler. The language expected in this
|
||
compile command is the language currently set in GDB. */
|
||
|
||
static void
|
||
compile_print_command (const char *arg, int from_tty)
|
||
{
|
||
enum compile_i_scope_types scope = COMPILE_I_PRINT_ADDRESS_SCOPE;
|
||
value_print_options print_opts;
|
||
|
||
scoped_restore save_async = make_scoped_restore (¤t_ui->async, 0);
|
||
|
||
get_user_print_options (&print_opts);
|
||
/* Override global settings with explicit options, if any. */
|
||
auto group = make_value_print_options_def_group (&print_opts);
|
||
gdb::option::process_options
|
||
(&arg, gdb::option::PROCESS_OPTIONS_REQUIRE_DELIMITER, group);
|
||
|
||
print_command_parse_format (&arg, "compile print", &print_opts);
|
||
|
||
/* Passing &PRINT_OPTS as SCOPE_DATA is safe as do_module_cleanup
|
||
will not touch the stale pointer if compile_object_run has
|
||
already quit. */
|
||
|
||
if (arg && *arg)
|
||
eval_compile_command (NULL, arg, scope, &print_opts);
|
||
else
|
||
{
|
||
counted_command_line l = get_command_line (compile_control, "");
|
||
|
||
l->control_u.compile.scope = scope;
|
||
l->control_u.compile.scope_data = &print_opts;
|
||
execute_control_command_untraced (l.get ());
|
||
}
|
||
}
|
||
|
||
/* A cleanup function to remove a directory and all its contents. */
|
||
|
||
static void
|
||
do_rmdir (void *arg)
|
||
{
|
||
const char *dir = (const char *) arg;
|
||
char *zap;
|
||
int wstat;
|
||
|
||
gdb_assert (startswith (dir, TMP_PREFIX));
|
||
zap = concat ("rm -rf ", dir, (char *) NULL);
|
||
wstat = system (zap);
|
||
if (wstat == -1 || !WIFEXITED (wstat) || WEXITSTATUS (wstat) != 0)
|
||
warning (_("Could not remove temporary directory %s"), dir);
|
||
XDELETEVEC (zap);
|
||
}
|
||
|
||
/* Return the name of the temporary directory to use for .o files, and
|
||
arrange for the directory to be removed at shutdown. */
|
||
|
||
static const char *
|
||
get_compile_file_tempdir (void)
|
||
{
|
||
static char *tempdir_name;
|
||
|
||
#define TEMPLATE TMP_PREFIX "XXXXXX"
|
||
char tname[sizeof (TEMPLATE)];
|
||
|
||
if (tempdir_name != NULL)
|
||
return tempdir_name;
|
||
|
||
strcpy (tname, TEMPLATE);
|
||
#undef TEMPLATE
|
||
tempdir_name = mkdtemp (tname);
|
||
if (tempdir_name == NULL)
|
||
perror_with_name (_("Could not make temporary directory"));
|
||
|
||
tempdir_name = xstrdup (tempdir_name);
|
||
make_final_cleanup (do_rmdir, tempdir_name);
|
||
return tempdir_name;
|
||
}
|
||
|
||
/* Compute the names of source and object files to use. */
|
||
|
||
static compile_file_names
|
||
get_new_file_names ()
|
||
{
|
||
static int seq;
|
||
const char *dir = get_compile_file_tempdir ();
|
||
|
||
++seq;
|
||
|
||
return compile_file_names (string_printf ("%s%sout%d.c",
|
||
dir, SLASH_STRING, seq),
|
||
string_printf ("%s%sout%d.o",
|
||
dir, SLASH_STRING, seq));
|
||
}
|
||
|
||
/* Get the block and PC at which to evaluate an expression. */
|
||
|
||
static const struct block *
|
||
get_expr_block_and_pc (CORE_ADDR *pc)
|
||
{
|
||
const struct block *block = get_selected_block (pc);
|
||
|
||
if (block == NULL)
|
||
{
|
||
struct symtab_and_line cursal = get_current_source_symtab_and_line ();
|
||
|
||
if (cursal.symtab)
|
||
block = BLOCKVECTOR_BLOCK (SYMTAB_BLOCKVECTOR (cursal.symtab),
|
||
STATIC_BLOCK);
|
||
if (block != NULL)
|
||
*pc = BLOCK_ENTRY_PC (block);
|
||
}
|
||
else
|
||
*pc = BLOCK_ENTRY_PC (block);
|
||
|
||
return block;
|
||
}
|
||
|
||
/* String for 'set compile-args' and 'show compile-args'. */
|
||
static char *compile_args;
|
||
|
||
/* Parsed form of COMPILE_ARGS. */
|
||
static gdb_argv compile_args_argv;
|
||
|
||
/* Implement 'set compile-args'. */
|
||
|
||
static void
|
||
set_compile_args (const char *args, int from_tty, struct cmd_list_element *c)
|
||
{
|
||
compile_args_argv = gdb_argv (compile_args);
|
||
}
|
||
|
||
/* Implement 'show compile-args'. */
|
||
|
||
static void
|
||
show_compile_args (struct ui_file *file, int from_tty,
|
||
struct cmd_list_element *c, const char *value)
|
||
{
|
||
fprintf_filtered (file, _("Compile command command-line arguments "
|
||
"are \"%s\".\n"),
|
||
value);
|
||
}
|
||
|
||
/* String for 'set compile-gcc' and 'show compile-gcc'. */
|
||
static char *compile_gcc;
|
||
|
||
/* Implement 'show compile-gcc'. */
|
||
|
||
static void
|
||
show_compile_gcc (struct ui_file *file, int from_tty,
|
||
struct cmd_list_element *c, const char *value)
|
||
{
|
||
fprintf_filtered (file, _("Compile command GCC driver filename is \"%s\".\n"),
|
||
value);
|
||
}
|
||
|
||
/* Return DW_AT_producer parsed for get_selected_frame () (if any).
|
||
Return NULL otherwise.
|
||
|
||
GCC already filters its command-line arguments only for the suitable ones to
|
||
put into DW_AT_producer - see GCC function gen_producer_string. */
|
||
|
||
static const char *
|
||
get_selected_pc_producer_options (void)
|
||
{
|
||
CORE_ADDR pc = get_frame_pc (get_selected_frame (NULL));
|
||
struct compunit_symtab *symtab = find_pc_compunit_symtab (pc);
|
||
const char *cs;
|
||
|
||
if (symtab == NULL || symtab->producer == NULL
|
||
|| !startswith (symtab->producer, "GNU "))
|
||
return NULL;
|
||
|
||
cs = symtab->producer;
|
||
while (*cs != 0 && *cs != '-')
|
||
cs = skip_spaces (skip_to_space (cs));
|
||
if (*cs != '-')
|
||
return NULL;
|
||
return cs;
|
||
}
|
||
|
||
/* Filter out unwanted options from ARGV. */
|
||
|
||
static void
|
||
filter_args (char **argv)
|
||
{
|
||
char **destv;
|
||
|
||
for (destv = argv; *argv != NULL; argv++)
|
||
{
|
||
/* -fpreprocessed may get in commonly from ccache. */
|
||
if (strcmp (*argv, "-fpreprocessed") == 0)
|
||
{
|
||
xfree (*argv);
|
||
continue;
|
||
}
|
||
*destv++ = *argv;
|
||
}
|
||
*destv = NULL;
|
||
}
|
||
|
||
/* Produce final vector of GCC compilation options.
|
||
|
||
The first element of the combined argument vector are arguments
|
||
relating to the target size ("-m64", "-m32" etc.). These are
|
||
sourced from the inferior's architecture.
|
||
|
||
The second element of the combined argument vector are arguments
|
||
stored in the inferior DW_AT_producer section. If these are stored
|
||
in the inferior (there is no guarantee that they are), they are
|
||
added to the vector.
|
||
|
||
The third element of the combined argument vector are argument
|
||
supplied by the language implementation provided by
|
||
compile-{lang}-support. These contain language specific arguments.
|
||
|
||
The final element of the combined argument vector are arguments
|
||
supplied by the "set compile-args" command. These are always
|
||
appended last so as to override any of the arguments automatically
|
||
generated above. */
|
||
|
||
static gdb_argv
|
||
get_args (const compile_instance *compiler, struct gdbarch *gdbarch)
|
||
{
|
||
const char *cs_producer_options;
|
||
gdb_argv result;
|
||
|
||
std::string gcc_options = gdbarch_gcc_target_options (gdbarch);
|
||
|
||
/* Make sure we have a non-empty set of options, otherwise GCC will
|
||
error out trying to look for a filename that is an empty string. */
|
||
if (!gcc_options.empty ())
|
||
result = gdb_argv (gcc_options.c_str ());
|
||
|
||
cs_producer_options = get_selected_pc_producer_options ();
|
||
if (cs_producer_options != NULL)
|
||
{
|
||
gdb_argv argv_producer (cs_producer_options);
|
||
filter_args (argv_producer.get ());
|
||
|
||
result.append (std::move (argv_producer));
|
||
}
|
||
|
||
result.append (gdb_argv (compiler->gcc_target_options ().c_str ()));
|
||
result.append (compile_args_argv);
|
||
|
||
return result;
|
||
}
|
||
|
||
/* A helper function suitable for use as the "print_callback" in the
|
||
compiler object. */
|
||
|
||
static void
|
||
print_callback (void *ignore, const char *message)
|
||
{
|
||
fputs_filtered (message, gdb_stderr);
|
||
}
|
||
|
||
/* Process the compilation request. On success it returns the object
|
||
and source file names. On an error condition, error () is
|
||
called. */
|
||
|
||
static compile_file_names
|
||
compile_to_object (struct command_line *cmd, const char *cmd_string,
|
||
enum compile_i_scope_types scope)
|
||
{
|
||
const struct block *expr_block;
|
||
CORE_ADDR trash_pc, expr_pc;
|
||
int ok;
|
||
struct gdbarch *gdbarch = get_current_arch ();
|
||
std::string triplet_rx;
|
||
|
||
if (!target_has_execution ())
|
||
error (_("The program must be running for the compile command to "\
|
||
"work."));
|
||
|
||
expr_block = get_expr_block_and_pc (&trash_pc);
|
||
expr_pc = get_frame_address_in_block (get_selected_frame (NULL));
|
||
|
||
/* Set up instance and context for the compiler. */
|
||
std::unique_ptr<compile_instance> compiler
|
||
= current_language->get_compile_instance ();
|
||
if (compiler == nullptr)
|
||
error (_("No compiler support for language %s."),
|
||
current_language->name ());
|
||
compiler->set_print_callback (print_callback, NULL);
|
||
compiler->set_scope (scope);
|
||
compiler->set_block (expr_block);
|
||
|
||
/* From the provided expression, build a scope to pass to the
|
||
compiler. */
|
||
|
||
string_file input_buf;
|
||
const char *input;
|
||
|
||
if (cmd != NULL)
|
||
{
|
||
struct command_line *iter;
|
||
|
||
for (iter = cmd->body_list_0.get (); iter; iter = iter->next)
|
||
{
|
||
input_buf.puts (iter->line);
|
||
input_buf.puts ("\n");
|
||
}
|
||
|
||
input = input_buf.c_str ();
|
||
}
|
||
else if (cmd_string != NULL)
|
||
input = cmd_string;
|
||
else
|
||
error (_("Neither a simple expression, or a multi-line specified."));
|
||
|
||
std::string code
|
||
= current_language->compute_program (compiler.get (), input, gdbarch,
|
||
expr_block, expr_pc);
|
||
if (compile_debug)
|
||
fprintf_unfiltered (gdb_stdlog, "debug output:\n\n%s", code.c_str ());
|
||
|
||
compiler->set_verbose (compile_debug);
|
||
|
||
if (compile_gcc[0] != 0)
|
||
{
|
||
if (compiler->version () < GCC_FE_VERSION_1)
|
||
error (_("Command 'set compile-gcc' requires GCC version 6 or higher "
|
||
"(libcc1 interface version 1 or higher)"));
|
||
|
||
compiler->set_driver_filename (compile_gcc);
|
||
}
|
||
else
|
||
{
|
||
const char *os_rx = osabi_triplet_regexp (gdbarch_osabi (gdbarch));
|
||
const char *arch_rx = gdbarch_gnu_triplet_regexp (gdbarch);
|
||
|
||
/* Allow triplets with or without vendor set. */
|
||
triplet_rx = std::string (arch_rx) + "(-[^-]*)?-";
|
||
if (os_rx != nullptr)
|
||
triplet_rx += os_rx;
|
||
compiler->set_triplet_regexp (triplet_rx.c_str ());
|
||
}
|
||
|
||
/* Set compiler command-line arguments. */
|
||
gdb_argv argv_holder = get_args (compiler.get (), gdbarch);
|
||
int argc = argv_holder.count ();
|
||
char **argv = argv_holder.get ();
|
||
|
||
gdb::unique_xmalloc_ptr<char> error_message
|
||
= compiler->set_arguments (argc, argv, triplet_rx.c_str ());
|
||
|
||
if (error_message != NULL)
|
||
error ("%s", error_message.get ());
|
||
|
||
if (compile_debug)
|
||
{
|
||
int argi;
|
||
|
||
fprintf_unfiltered (gdb_stdlog, "Passing %d compiler options:\n", argc);
|
||
for (argi = 0; argi < argc; argi++)
|
||
fprintf_unfiltered (gdb_stdlog, "Compiler option %d: <%s>\n",
|
||
argi, argv[argi]);
|
||
}
|
||
|
||
compile_file_names fnames = get_new_file_names ();
|
||
|
||
gdb::optional<gdb::unlinker> source_remover;
|
||
|
||
{
|
||
gdb_file_up src = gdb_fopen_cloexec (fnames.source_file (), "w");
|
||
if (src == NULL)
|
||
perror_with_name (_("Could not open source file for writing"));
|
||
|
||
source_remover.emplace (fnames.source_file ());
|
||
|
||
if (fputs (code.c_str (), src.get ()) == EOF)
|
||
perror_with_name (_("Could not write to source file"));
|
||
}
|
||
|
||
if (compile_debug)
|
||
fprintf_unfiltered (gdb_stdlog, "source file produced: %s\n\n",
|
||
fnames.source_file ());
|
||
|
||
/* If we don't do this, then GDB simply exits
|
||
when the compiler dies. */
|
||
scoped_ignore_sigpipe ignore_sigpipe;
|
||
|
||
/* Call the compiler and start the compilation process. */
|
||
compiler->set_source_file (fnames.source_file ());
|
||
ok = compiler->compile (fnames.object_file (), compile_debug);
|
||
if (!ok)
|
||
error (_("Compilation failed."));
|
||
|
||
if (compile_debug)
|
||
fprintf_unfiltered (gdb_stdlog, "object file produced: %s\n\n",
|
||
fnames.object_file ());
|
||
|
||
/* Keep the source file. */
|
||
source_remover->keep ();
|
||
return fnames;
|
||
}
|
||
|
||
/* The "compile" prefix command. */
|
||
|
||
static void
|
||
compile_command (const char *args, int from_tty)
|
||
{
|
||
/* If a sub-command is not specified to the compile prefix command,
|
||
assume it is a direct code compilation. */
|
||
compile_code_command (args, from_tty);
|
||
}
|
||
|
||
/* See compile.h. */
|
||
|
||
void
|
||
eval_compile_command (struct command_line *cmd, const char *cmd_string,
|
||
enum compile_i_scope_types scope, void *scope_data)
|
||
{
|
||
compile_file_names fnames = compile_to_object (cmd, cmd_string, scope);
|
||
|
||
gdb::unlinker object_remover (fnames.object_file ());
|
||
gdb::unlinker source_remover (fnames.source_file ());
|
||
|
||
compile_module_up compile_module = compile_object_load (fnames, scope,
|
||
scope_data);
|
||
if (compile_module == NULL)
|
||
{
|
||
gdb_assert (scope == COMPILE_I_PRINT_ADDRESS_SCOPE);
|
||
eval_compile_command (cmd, cmd_string,
|
||
COMPILE_I_PRINT_VALUE_SCOPE, scope_data);
|
||
return;
|
||
}
|
||
|
||
/* Keep the files. */
|
||
source_remover.keep ();
|
||
object_remover.keep ();
|
||
|
||
compile_object_run (std::move (compile_module));
|
||
}
|
||
|
||
/* See compile/compile-internal.h. */
|
||
|
||
std::string
|
||
compile_register_name_mangled (struct gdbarch *gdbarch, int regnum)
|
||
{
|
||
const char *regname = gdbarch_register_name (gdbarch, regnum);
|
||
|
||
return string_printf ("__%s", regname);
|
||
}
|
||
|
||
/* See compile/compile-internal.h. */
|
||
|
||
int
|
||
compile_register_name_demangle (struct gdbarch *gdbarch,
|
||
const char *regname)
|
||
{
|
||
int regnum;
|
||
|
||
if (regname[0] != '_' || regname[1] != '_')
|
||
error (_("Invalid register name \"%s\"."), regname);
|
||
regname += 2;
|
||
|
||
for (regnum = 0; regnum < gdbarch_num_regs (gdbarch); regnum++)
|
||
if (strcmp (regname, gdbarch_register_name (gdbarch, regnum)) == 0)
|
||
return regnum;
|
||
|
||
error (_("Cannot find gdbarch register \"%s\"."), regname);
|
||
}
|
||
|
||
/* Forwards to the plug-in. */
|
||
|
||
#define FORWARD(OP,...) (m_gcc_fe->ops->OP (m_gcc_fe, ##__VA_ARGS__))
|
||
|
||
/* See compile-internal.h. */
|
||
|
||
void
|
||
compile_instance::set_print_callback
|
||
(void (*print_function) (void *, const char *), void *datum)
|
||
{
|
||
FORWARD (set_print_callback, print_function, datum);
|
||
}
|
||
|
||
/* See compile-internal.h. */
|
||
|
||
unsigned int
|
||
compile_instance::version () const
|
||
{
|
||
return m_gcc_fe->ops->version;
|
||
}
|
||
|
||
/* See compile-internal.h. */
|
||
|
||
void
|
||
compile_instance::set_verbose (int level)
|
||
{
|
||
if (version () >= GCC_FE_VERSION_1)
|
||
FORWARD (set_verbose, level);
|
||
}
|
||
|
||
/* See compile-internal.h. */
|
||
|
||
void
|
||
compile_instance::set_driver_filename (const char *filename)
|
||
{
|
||
if (version () >= GCC_FE_VERSION_1)
|
||
FORWARD (set_driver_filename, filename);
|
||
}
|
||
|
||
/* See compile-internal.h. */
|
||
|
||
void
|
||
compile_instance::set_triplet_regexp (const char *regexp)
|
||
{
|
||
if (version () >= GCC_FE_VERSION_1)
|
||
FORWARD (set_triplet_regexp, regexp);
|
||
}
|
||
|
||
/* See compile-internal.h. */
|
||
|
||
gdb::unique_xmalloc_ptr<char>
|
||
compile_instance::set_arguments (int argc, char **argv, const char *regexp)
|
||
{
|
||
if (version () >= GCC_FE_VERSION_1)
|
||
return gdb::unique_xmalloc_ptr<char> (FORWARD (set_arguments, argc, argv));
|
||
else
|
||
return gdb::unique_xmalloc_ptr<char> (FORWARD (set_arguments_v0, regexp,
|
||
argc, argv));
|
||
}
|
||
|
||
/* See compile-internal.h. */
|
||
|
||
void
|
||
compile_instance::set_source_file (const char *filename)
|
||
{
|
||
FORWARD (set_source_file, filename);
|
||
}
|
||
|
||
/* See compile-internal.h. */
|
||
|
||
bool
|
||
compile_instance::compile (const char *filename, int verbose_level)
|
||
{
|
||
if (version () >= GCC_FE_VERSION_1)
|
||
return FORWARD (compile, filename);
|
||
else
|
||
return FORWARD (compile_v0, filename, verbose_level);
|
||
}
|
||
|
||
#undef FORWARD
|
||
|
||
/* See compile.h. */
|
||
cmd_list_element *compile_cmd_element = nullptr;
|
||
|
||
void _initialize_compile ();
|
||
void
|
||
_initialize_compile ()
|
||
{
|
||
struct cmd_list_element *c = NULL;
|
||
|
||
compile_cmd_element = add_prefix_cmd ("compile", class_obscure,
|
||
compile_command, _("\
|
||
Command to compile source code and inject it into the inferior."),
|
||
&compile_command_list, 1, &cmdlist);
|
||
add_com_alias ("expression", compile_cmd_element, class_obscure, 0);
|
||
|
||
const auto compile_opts = make_compile_options_def_group (nullptr);
|
||
|
||
static const std::string compile_code_help
|
||
= gdb::option::build_help (_("\
|
||
Compile, inject, and execute code.\n\
|
||
\n\
|
||
Usage: compile code [OPTION]... [CODE]\n\
|
||
\n\
|
||
Options:\n\
|
||
%OPTIONS%\n\
|
||
\n\
|
||
The source code may be specified as a simple one line expression, e.g.:\n\
|
||
\n\
|
||
compile code printf(\"Hello world\\n\");\n\
|
||
\n\
|
||
Alternatively, you can type a multiline expression by invoking\n\
|
||
this command with no argument. GDB will then prompt for the\n\
|
||
expression interactively; type a line containing \"end\" to\n\
|
||
indicate the end of the expression."),
|
||
compile_opts);
|
||
|
||
c = add_cmd ("code", class_obscure, compile_code_command,
|
||
compile_code_help.c_str (),
|
||
&compile_command_list);
|
||
set_cmd_completer_handle_brkchars (c, compile_code_command_completer);
|
||
|
||
static const std::string compile_file_help
|
||
= gdb::option::build_help (_("\
|
||
Evaluate a file containing source code.\n\
|
||
\n\
|
||
Usage: compile file [OPTION].. [FILENAME]\n\
|
||
\n\
|
||
Options:\n\
|
||
%OPTIONS%"),
|
||
compile_opts);
|
||
|
||
c = add_cmd ("file", class_obscure, compile_file_command,
|
||
compile_file_help.c_str (),
|
||
&compile_command_list);
|
||
set_cmd_completer_handle_brkchars (c, compile_file_command_completer);
|
||
|
||
const auto compile_print_opts = make_value_print_options_def_group (nullptr);
|
||
|
||
static const std::string compile_print_help
|
||
= gdb::option::build_help (_("\
|
||
Evaluate EXPR by using the compiler and print result.\n\
|
||
\n\
|
||
Usage: compile print [[OPTION]... --] [/FMT] [EXPR]\n\
|
||
\n\
|
||
Options:\n\
|
||
%OPTIONS%\n\
|
||
\n\
|
||
Note: because this command accepts arbitrary expressions, if you\n\
|
||
specify any command option, you must use a double dash (\"--\")\n\
|
||
to mark the end of option processing. E.g.: \"compile print -o -- myobj\".\n\
|
||
\n\
|
||
The expression may be specified on the same line as the command, e.g.:\n\
|
||
\n\
|
||
compile print i\n\
|
||
\n\
|
||
Alternatively, you can type a multiline expression by invoking\n\
|
||
this command with no argument. GDB will then prompt for the\n\
|
||
expression interactively; type a line containing \"end\" to\n\
|
||
indicate the end of the expression.\n\
|
||
\n\
|
||
EXPR may be preceded with /FMT, where FMT is a format letter\n\
|
||
but no count or size letter (see \"x\" command)."),
|
||
compile_print_opts);
|
||
|
||
c = add_cmd ("print", class_obscure, compile_print_command,
|
||
compile_print_help.c_str (),
|
||
&compile_command_list);
|
||
set_cmd_completer_handle_brkchars (c, print_command_completer);
|
||
|
||
add_setshow_boolean_cmd ("compile", class_maintenance, &compile_debug, _("\
|
||
Set compile command debugging."), _("\
|
||
Show compile command debugging."), _("\
|
||
When on, compile command debugging is enabled."),
|
||
NULL, show_compile_debug,
|
||
&setdebuglist, &showdebuglist);
|
||
|
||
add_setshow_string_cmd ("compile-args", class_support,
|
||
&compile_args,
|
||
_("Set compile command GCC command-line arguments."),
|
||
_("Show compile command GCC command-line arguments."),
|
||
_("\
|
||
Use options like -I (include file directory) or ABI settings.\n\
|
||
String quoting is parsed like in shell, for example:\n\
|
||
-mno-align-double \"-I/dir with a space/include\""),
|
||
set_compile_args, show_compile_args, &setlist, &showlist);
|
||
|
||
/* Override flags possibly coming from DW_AT_producer. */
|
||
compile_args = xstrdup ("-O0 -gdwarf-4"
|
||
/* We use -fPIE Otherwise GDB would need to reserve space large enough for
|
||
any object file in the inferior in advance to get the final address when
|
||
to link the object file to and additionally the default system linker
|
||
script would need to be modified so that one can specify there the
|
||
absolute target address.
|
||
-fPIC is not used at is would require from GDB to generate .got. */
|
||
" -fPIE"
|
||
/* We want warnings, except for some commonly happening for GDB commands. */
|
||
" -Wall "
|
||
" -Wno-unused-but-set-variable"
|
||
" -Wno-unused-variable"
|
||
/* Override CU's possible -fstack-protector-strong. */
|
||
" -fno-stack-protector"
|
||
);
|
||
set_compile_args (compile_args, 0, NULL);
|
||
|
||
add_setshow_optional_filename_cmd ("compile-gcc", class_support,
|
||
&compile_gcc,
|
||
_("Set compile command "
|
||
"GCC driver filename."),
|
||
_("Show compile command "
|
||
"GCC driver filename."),
|
||
_("\
|
||
It should be absolute filename of the gcc executable.\n\
|
||
If empty the default target triplet will be searched in $PATH."),
|
||
NULL, show_compile_gcc, &setlist,
|
||
&showlist);
|
||
compile_gcc = xstrdup ("");
|
||
}
|