mirror of
https://sourceware.org/git/binutils-gdb.git
synced 2024-12-21 04:42:53 +08:00
491144b5e2
This is for add_setshow_boolean_cmd as well as the gdb::option interface. gdb/ChangeLog: 2019-09-17 Christian Biesinger <cbiesinger@google.com> * ada-lang.c (ada_ignore_descriptive_types_p): Change to bool. (print_signatures): Likewise. (trust_pad_over_xvs): Likewise. * arch/aarch64-insn.c (aarch64_debug): Likewise. * arch/aarch64-insn.h (aarch64_debug): Likewise. * arm-linux-nat.c (arm_apcs_32): Likewise. * arm-linux-tdep.c (arm_apcs_32): Likewise. * arm-nbsd-nat.c (arm_apcs_32): Likewise. * arm-tdep.c (arm_debug): Likewise. (arm_apcs_32): Likewise. * auto-load.c (debug_auto_load): Likewise. (auto_load_gdb_scripts): Likewise. (global_auto_load): Likewise. (auto_load_local_gdbinit): Likewise. (auto_load_local_gdbinit_loaded): Likewise. * auto-load.h (global_auto_load): Likewise. (auto_load_local_gdbinit): Likewise. (auto_load_local_gdbinit_loaded): Likewise. * breakpoint.c (disconnected_dprintf): Likewise. (breakpoint_proceeded): Likewise. (automatic_hardware_breakpoints): Likewise. (always_inserted_mode): Likewise. (target_exact_watchpoints): Likewise. (_initialize_breakpoint): Update. * breakpoint.h (target_exact_watchpoints): Change to bool. * btrace.c (maint_btrace_pt_skip_pad): Likewise. * cli/cli-cmds.c (trace_commands): Likewise. * cli/cli-cmds.h (trace_commands): Likewise. * cli/cli-decode.c (add_setshow_boolean_cmd): Change int* argument to bool*. * cli/cli-logging.c (logging_overwrite): Change to bool. (logging_redirect): Likewise. (debug_redirect): Likewise. * cli/cli-option.h (option_def) <boolean>: Change return type to bool*. (struct boolean_option_def) <get_var_address_cb_>: Change return type to bool. <boolean_option_def>: Update. (struct flag_option_def): Change default type of Context to bool from int. <flag_option_def>: Change return type of var_address_cb_ to bool*. * cli/cli-setshow.c (do_set_command): Cast to bool* instead of int*. (get_setshow_command_value_string): Likewise. * cli/cli-style.c (cli_styling): Change to bool. (source_styling): Likewise. * cli/cli-style.h (source_styling): Likewise. (cli_styling): Likewise. * cli/cli-utils.h (struct qcs_flags) <quiet, cont, silent>: Change to bool. * command.h (var_types): Update comment. (add_setshow_boolean_cmd): Change int* var argument to bool*. * compile/compile-cplus-types.c (debug_compile_cplus_types): Change to bool. (debug_compile_cplus_scopes): Likewise. * compile/compile-internal.h (compile_debug): Likewise. * compile/compile.c (compile_debug): Likewise. (struct compile_options) <raw>: Likewise. * cp-support.c (catch_demangler_crashes): Likewise. * cris-tdep.c (usr_cmd_cris_version_valid): Likewise. (usr_cmd_cris_dwarf2_cfi): Likewise. * csky-tdep.c (csky_debug): Likewise. * darwin-nat.c (enable_mach_exceptions): Likewise. * dcache.c (dcache_enabled_p): Likewise. * defs.h (info_verbose): Likewise. * demangle.c (demangle): Likewise. (asm_demangle): Likewise. * dwarf-index-cache.c (debug_index_cache): Likewise. * dwarf2-frame.c (dwarf2_frame_unwinders_enabled_p): Likewise. * dwarf2-frame.h (dwarf2_frame_unwinders_enabled_p): Likewise. * dwarf2read.c (check_physname): Likewise. (use_deprecated_index_sections): Likewise. (dwarf_always_disassemble): Likewise. * eval.c (overload_resolution): Likewise. * event-top.c (set_editing_cmd_var): Likewise. (exec_done_display_p): Likewise. * event-top.h (set_editing_cmd_var): Likewise. (exec_done_display_p): Likewise. * exec.c (write_files): Likewise. * fbsd-nat.c (debug_fbsd_lwp): Likewise (debug_fbsd_nat): Likewise. * frame.h (struct frame_print_options) <print_raw_frame_arguments>: Likewise. (struct set_backtrace_options) <backtrace_past_main>: Likewise. <backtrace_past_entry> Likewise. * gdb-demangle.h (demangle): Likewise. (asm_demangle): Likewise. * gdb_bfd.c (bfd_sharing): Likewise. * gdbcore.h (write_files): Likewise. * gdbsupport/common-debug.c (show_debug_regs): Likewise. * gdbsupport/common-debug.h (show_debug_regs): Likewise. * gdbthread.h (print_thread_events): Likewise. * gdbtypes.c (opaque_type_resolution): Likewise. (strict_type_checking): Likewise. * gnu-nat.c (gnu_debug_flag): Likewise. * guile/scm-auto-load.c (auto_load_guile_scripts): Likewise. * guile/scm-param.c (pascm_variable): Add boolval. (add_setshow_generic): Update. (pascm_param_value): Update. (pascm_set_param_value_x): Update. * hppa-tdep.c (hppa_debug): Change to bool.. * infcall.c (may_call_functions_p): Likewise. (coerce_float_to_double_p): Likewise. (unwind_on_signal_p): Likewise. (unwind_on_terminating_exception_p): Likewise. * infcmd.c (startup_with_shell): Likewise. * inferior.c (print_inferior_events): Likewise. * inferior.h (startup_with_shell): Likewise. (print_inferior_events): Likewise. * infrun.c (step_stop_if_no_debug): Likewise. (detach_fork): Likewise. (debug_displaced): Likewise. (disable_randomization): Likewise. (non_stop): Likewise. (non_stop_1): Likewise. (observer_mode): Likewise. (observer_mode_1): Likewise. (set_observer_mode): Update. (sched_multi): Change to bool. * infrun.h (debug_displaced): Likewise. (sched_multi): Likewise. (step_stop_if_no_debug): Likewise. (non_stop): Likewise. (disable_randomization): Likewise. * linux-tdep.c (use_coredump_filter): Likewise. (dump_excluded_mappings): Likewise. * linux-thread-db.c (auto_load_thread_db): Likewise. (check_thread_db_on_load): Likewise. * main.c (captured_main_1): Update. * maint-test-options.c (struct test_options_opts) <flag_opt, xx1_opt, xx2_opt, boolean_opt>: Change to bool. * maint-test-settings.c (maintenance_test_settings_boolean): Likewise. * maint.c (maintenance_profile_p): Likewise. (per_command_time): Likewise. (per_command_space): Likewise. (per_command_symtab): Likewise. * memattr.c (inaccessible_by_default): Likewise. * mi/mi-main.c (mi_async): Likewise. (mi_async_1): Likewise. * mips-tdep.c (mips64_transfers_32bit_regs_p): Likewise. * nat/fork-inferior.h (startup_with_shell): Likewise. * nat/linux-namespaces.c (debug_linux_namespaces): Likewise. * nat/linux-namespaces.h (debug_linux_namespaces): Likewise. * nios2-tdep.c (nios2_debug): Likewise. * or1k-tdep.c (or1k_debug): Likewise. * parse.c (parser_debug): Likewise. * parser-defs.h (parser_debug): Likewise. * printcmd.c (print_symbol_filename): Likewise. * proc-api.c (procfs_trace): Likewise. * python/py-auto-load.c (auto_load_python_scripts): Likewise. * python/py-param.c (union parmpy_variable): Add "bool boolval" field. (set_parameter_value): Update. (add_setshow_generic): Update. * python/py-value.c (copy_py_bool_obj): Change argument from int* to bool*. * python/python.c (gdbpy_parameter_value): Cast to bool* instead of int*. * ravenscar-thread.c (ravenscar_task_support): Change to bool. * record-btrace.c (record_btrace_target::store_registers): Update. * record-full.c (record_full_memory_query): Change to bool. (record_full_stop_at_limit): Likewise. * record-full.h (record_full_memory_query): Likewise. * remote-notif.c (notif_debug): Likewise. * remote-notif.h (notif_debug): Likewise. * remote.c (use_range_stepping): Likewise. (interrupt_on_connect): Likewise. (remote_break): Likewise. * ser-tcp.c (tcp_auto_retry): Likewise. * ser-unix.c (serial_hwflow): Likewise. * skip.c (debug_skip): Likewise. * solib-aix.c (solib_aix_debug): Likewise. * spu-tdep.c (spu_stop_on_load_p): Likewise. (spu_auto_flush_cache_p): Likewise. * stack.c (struct backtrace_cmd_options) <full, no_filters, hide>: Likewise. (struct info_print_options) <quiet>: Likewise. * symfile-debug.c (debug_symfile): Likewise. * symfile.c (auto_solib_add): Likewise. (separate_debug_file_debug): Likewise. * symfile.h (auto_solib_add): Likewise. (separate_debug_file_debug): Likewise. * symtab.c (basenames_may_differ): Likewise. (struct filename_partial_match_opts) <dirname, basename>: Likewise. (struct info_print_options) <quiet, exclude_minsyms>: Likewise. (struct info_types_options) <quiet>: Likewise. * symtab.h (demangle): Likewise. (basenames_may_differ): Likewise. * target-dcache.c (stack_cache_enabled_1): Likewise. (code_cache_enabled_1): Likewise. * target.c (trust_readonly): Likewise. (may_write_registers): Likewise. (may_write_memory): Likewise. (may_insert_breakpoints): Likewise. (may_insert_tracepoints): Likewise. (may_insert_fast_tracepoints): Likewise. (may_stop): Likewise. (auto_connect_native_target): Likewise. (target_stop_and_wait): Update. (target_async_permitted): Change to bool. (target_async_permitted_1): Likewise. (may_write_registers_1): Likewise. (may_write_memory_1): Likewise. (may_insert_breakpoints_1): Likewise. (may_insert_tracepoints_1): Likewise. (may_insert_fast_tracepoints_1): Likewise. (may_stop_1): Likewise. * target.h (target_async_permitted): Likewise. (may_write_registers): Likewise. (may_write_memory): Likewise. (may_insert_breakpoints): Likewise. (may_insert_tracepoints): Likewise. (may_insert_fast_tracepoints): Likewise. (may_stop): Likewise. * thread.c (struct info_threads_opts) <show_global_ids>: Likewise. (make_thread_apply_all_options_def_group): Change argument from int* to bool*. (thread_apply_all_command): Update. (print_thread_events): Change to bool. * top.c (confirm): Likewise. (command_editing_p): Likewise. (history_expansion_p): Likewise. (write_history_p): Likewise. (info_verbose): Likewise. * top.h (confirm): Likewise. (history_expansion_p): Likewise. * tracepoint.c (disconnected_tracing): Likewise. (circular_trace_buffer): Likewise. * typeprint.c (print_methods): Likewise. (print_typedefs): Likewise. * utils.c (debug_timestamp): Likewise. (sevenbit_strings): Likewise. (pagination_enabled): Likewise. * utils.h (sevenbit_strings): Likewise. (pagination_enabled): Likewise. * valops.c (overload_resolution): Likewise. * valprint.h (struct value_print_options) <prettyformat_arrays, prettyformat_structs, vtblprint, unionprint, addressprint, objectprint, stop_print_at_null, print_array_indexes, deref_ref, static_field_print, pascal_static_field_print, raw, summary, symbol_print, finish_print>: Likewise. * windows-nat.c (new_console): Likewise. (cygwin_exceptions): Likewise. (new_group): Likewise. (debug_exec): Likewise. (debug_events): Likewise. (debug_memory): Likewise. (debug_exceptions): Likewise. (useshell): Likewise. * windows-tdep.c (maint_display_all_tib): Likewise. * xml-support.c (debug_xml): Likewise.
1093 lines
30 KiB
C
1093 lines
30 KiB
C
/* General Compile and inject code
|
||
|
||
Copyright (C) 2014-2019 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"
|
||
|
||
|
||
|
||
/* 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;
|
||
}
|
||
|
||
/* Call buildargv (via gdb_argv), set its result for S into *ARGVP but
|
||
calculate also the number of parsed arguments into *ARGCP. If
|
||
buildargv has returned NULL then *ARGCP is set to zero. */
|
||
|
||
static void
|
||
build_argc_argv (const char *s, int *argcp, char ***argvp)
|
||
{
|
||
gdb_argv args (s);
|
||
|
||
*argcp = args.count ();
|
||
*argvp = args.release ();
|
||
}
|
||
|
||
/* String for 'set compile-args' and 'show compile-args'. */
|
||
static char *compile_args;
|
||
|
||
/* Parsed form of COMPILE_ARGS. COMPILE_ARGS_ARGV is NULL terminated. */
|
||
static int compile_args_argc;
|
||
static char **compile_args_argv;
|
||
|
||
/* Implement 'set compile-args'. */
|
||
|
||
static void
|
||
set_compile_args (const char *args, int from_tty, struct cmd_list_element *c)
|
||
{
|
||
freeargv (compile_args_argv);
|
||
build_argc_argv (compile_args, &compile_args_argc, &compile_args_argv);
|
||
}
|
||
|
||
/* 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);
|
||
}
|
||
|
||
/* Append ARGC and ARGV (as parsed by build_argc_argv) to *ARGCP and *ARGVP.
|
||
ARGCP+ARGVP can be zero+NULL and also ARGC+ARGV can be zero+NULL. */
|
||
|
||
static void
|
||
append_args (int *argcp, char ***argvp, int argc, char **argv)
|
||
{
|
||
int argi;
|
||
|
||
*argvp = XRESIZEVEC (char *, *argvp, (*argcp + argc + 1));
|
||
|
||
for (argi = 0; argi < argc; argi++)
|
||
(*argvp)[(*argcp)++] = xstrdup (argv[argi]);
|
||
(*argvp)[(*argcp)] = NULL;
|
||
}
|
||
|
||
/* 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 *ARGCP and ARGV. */
|
||
|
||
static void
|
||
filter_args (int *argcp, char **argv)
|
||
{
|
||
char **destv;
|
||
|
||
for (destv = argv; *argv != NULL; argv++)
|
||
{
|
||
/* -fpreprocessed may get in commonly from ccache. */
|
||
if (strcmp (*argv, "-fpreprocessed") == 0)
|
||
{
|
||
xfree (*argv);
|
||
(*argcp)--;
|
||
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 void
|
||
get_args (const compile_instance *compiler, struct gdbarch *gdbarch,
|
||
int *argcp, char ***argvp)
|
||
{
|
||
const char *cs_producer_options;
|
||
int argc_compiler;
|
||
char **argv_compiler;
|
||
|
||
build_argc_argv (gdbarch_gcc_target_options (gdbarch),
|
||
argcp, argvp);
|
||
|
||
cs_producer_options = get_selected_pc_producer_options ();
|
||
if (cs_producer_options != NULL)
|
||
{
|
||
int argc_producer;
|
||
char **argv_producer;
|
||
|
||
build_argc_argv (cs_producer_options, &argc_producer, &argv_producer);
|
||
filter_args (&argc_producer, argv_producer);
|
||
append_args (argcp, argvp, argc_producer, argv_producer);
|
||
freeargv (argv_producer);
|
||
}
|
||
|
||
build_argc_argv (compiler->gcc_target_options ().c_str (),
|
||
&argc_compiler, &argv_compiler);
|
||
append_args (argcp, argvp, argc_compiler, argv_compiler);
|
||
freeargv (argv_compiler);
|
||
|
||
append_args (argcp, argvp, compile_args_argc, compile_args_argv);
|
||
}
|
||
|
||
/* 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 argc;
|
||
char **argv;
|
||
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. */
|
||
if (current_language->la_get_compile_instance == NULL)
|
||
error (_("No compiler support for language %s."),
|
||
current_language->la_name);
|
||
|
||
compile_instance *compiler_instance
|
||
= current_language->la_get_compile_instance ();
|
||
std::unique_ptr<compile_instance> compiler (compiler_instance);
|
||
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->la_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) + "(-[^-]*)?-" + os_rx;
|
||
compiler->set_triplet_regexp (triplet_rx.c_str ());
|
||
}
|
||
|
||
/* Set compiler command-line arguments. */
|
||
get_args (compiler.get (), gdbarch, &argc, &argv);
|
||
gdb_argv argv_holder (argv);
|
||
|
||
gdb::unique_xmalloc_ptr<char> error_message;
|
||
error_message.reset (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 ());
|
||
|
||
/* 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)
|
||
{
|
||
struct compile_module *compile_module;
|
||
|
||
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 = 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 (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. */
|
||
|
||
char *
|
||
compile_instance::set_arguments (int argc, char **argv, const char *regexp)
|
||
{
|
||
if (version () >= GCC_FE_VERSION_1)
|
||
return FORWARD (set_arguments, argc, argv);
|
||
else
|
||
return 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)
|
||
{
|
||
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, "compile ", 1, &cmdlist);
|
||
add_com_alias ("expression", "compile", 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 ("");
|
||
}
|