forked from Imagelibrary/binutils-gdb
This patch finalizes the C++ conversion of the ui-out subsystem, by
turning the ui_out and ui_out_impl structures into a single class
hierarchy. ui_out functions are turned into virtual methods of that new
class, so as a result there are a lot of call sites to update.
In the previous version of the patchset, there were separate ui_out and
ui_out_impl classes, but it wasn't really useful and added boilerplate.
In this version there is simply an ui_out base class that is
extended for CLI, TUI and MI.
It's a bit hard to maintain a ChangeLog for such a big patch, I did my
best but I'm sure there are some missing or outdated info in there...
gdb/ChangeLog:
* ui-out.h (ui_out_begin, ui_out_end, ui_out_table_header,
ui_out_table_body, ui_out_field_int, ui_out_field_fmt_int,
ui_out_field_core_addr, ui_out_field_string, ui_out_field_stream,
ui_out_field_fmt, ui_out_field_skip, ui_out_spaces, ui_out_text,
ui_out_message, ui_out_wrap_hint, ui_out_flush, ui_out_test_flags,
ui_out_query_field, ui_out_is_mi_like_p, ui_out_redirect):
Remove, replace with a method in class ui_out.
(table_begin_ftype): Remove, replace with pure virtual method in
class ui_out.
(table_body_ftype): Likewise.
(table_end_ftype): Likewise.
(table_header_ftype): Likewise.
(ui_out_begin_ftype): Likewise.
(ui_out_end_ftype): Likewise.
(field_int_ftype): Likewise.
(field_skip_ftype): Likewise.
(field_string_ftype): Likewise.
(field_fmt_ftype): Likewise.
(spaces_ftype): Likewise.
(text_ftype): Likewise.
(message_ftype): Likewise.
(wrap_hint_ftype): Likewise.
(flush_ftype): Likewise.
(redirect_ftype): Likewise.
(data_destroy_ftype): Likewise.
(struct ui_out_impl): Remove, replace with class ui_out.
(ui_out_new): Remove.
(class ui_out): New class.
* ui-out.c (struct ui_out): Remove, replaced with class ui_out.
(current_level): Remove, replace with ui_out method.
(push_level): Likewise.
(pop_level): Likewise.
(uo_table_begin, uo_table_body, uo_table_end, uo_table_header,
uo_begin, uo_end, uo_field_int, uo_field_skip, uo_field_fmt,
uo_spaces, uo_text, uo_message, uo_wrap_hint, uo_flush,
uo_redirect, uo_field_string): Remove.
(ui_out_table_begin): Replace with ...
(ui_out::table_begin): ... this.
(ui_out_table_body): Replace with ...
(ui_out::table_body): ... this.
(ui_out_table_end): Replace with ...
(ui_out::table_end): ... this.
(ui_out_table_header): Replace with ...
(ui_out::table_header): ... this.
(ui_out_begin): Replace with ...
(ui_out::begin): ... this.
(ui_out_end): Replace with ...
(ui_out::end): ... this.
(ui_out_field_int): Replace with ...
(ui_out::field_int): ... this.
(ui_out_field_fmt_int): Replace with ...
(ui_out::field_fmt_int): ... this.
(ui_out_field_core_addr): Replace with ...
(ui_out::field_core_addr): ... this.
(ui_out_field_stream): Replace with ...
(ui_out::field_stream): ... this.
(ui_out_field_skip): Replace with ...
(ui_out::field_skip): ... this.
(ui_out_field_string): Replace with ...
(ui_out::field_string): ... this.
(ui_out_field_fmt): Replace with ...
(ui_out::field_fmt): ... this.
(ui_out_spaces): Replace with ...
(ui_out::spaces): ... this.
(ui_out_text): Replace with ...
(ui_out::text): ... this.
(ui_out_message): Replace with ...
(ui_out::message): ... this.
(ui_out_wrap_hint): Replace with ...
(ui_out::wrap_hint): ... this.
(ui_out_flush): Replace with ...
(ui_out::flush): ... this.
(ui_out_redirect): Replace with ...
(ui_out::redirect): ... this.
(ui_out_test_flags): Replace with ...
(ui_out::test_flags): ... this.
(ui_out_is_mi_like_p): Replace with ...
(ui_out::is_mi_like_p): ... this.
(verify_field): Replace with ...
(ui_out::verify_field): ... this.
(ui_out_query_field): Replace with ...
(ui_out::query_table_field): ... this.
(ui_out_data): Remove.
(ui_out_new): Remove, replace with ...
(ui_out::ui_out): ... this constructor.
(do_cleanup_table_end, make_cleanup_ui_out_tuple_begin_end,
do_cleanup_end, make_cleanup_ui_out_tuple_begin_end,
make_cleanup_ui_out_list_begin_end): Update fallouts of struct
ui_out -> class ui_out change.
* cli-out.c (cli_out_data): Remove.
(cli_uiout_dtor): Remove.
(cli_table_begin): Replace with ...
(cli_ui_out::do_table_begin): ... this new method.
(cli_table_body): Replace with ...
(cli_ui_out::do_table_body): ... this new method.
(cli_table_end): Replace with ...
(cli_ui_out::do_table_end): ... this new method.
(cli_table_header): Replace with ...
(cli_ui_out::do_table_header): ... this new method.
(cli_begin): Replace with ...
(cli_ui_out::do_begin): ... this new method.
(cli_end): Replace with ...
(cli_ui_out::do_end): ... this new method.
(cli_field_int): Replace with ...
(cli_ui_out::do_field_int): ... this new method.
(cli_field_skip): Replace with ...
(cli_ui_out::do_field_skip): ... this new method.
(cli_field_string): Replace with ...
(cli_ui_out::do_field_string): ... this new method.
(cli_field_fmt): Replace with ...
(cli_ui_out::do_field_fmt): ... this new method.
(cli_spaces): Replace with ...
(cli_ui_out::do_spaces): ... this new method.
(cli_text): Replace with ...
(cli_ui_out::do_text): ... this new method.
(cli_message): Replace with ...
(cli_ui_out::do_message): ... this new method.
(cli_wrap_hint): Replace with ...
(cli_ui_out::do_wrap_hint): ... this new method.
(cli_flush): Replace with ...
(cli_ui_out::do_flush): ... this new method.
(cli_redirect): Replace with ...
(cli_ui_out::do_redirect): ... this new method.
(out_field_fmt): Replace with ...
(cli_ui_out::out_field_fmt): ... this new method.
(field_separator): Replace with ...
(cli_ui_out::field_separator): ... this new method.
(cli_out_set_stream): Replace with ...
(cli_ui_out::set_stream): ... this new method.
(cli_ui_out_impl): Remove.
(cli_out_data_ctor): Remove.
(cli_ui_out_impl::cli_ui_out_impl): New constructor.
(cli_ui_out_impl::~cli_ui_out_impl): New destructor.
(cli_out_new): Change return type to cli_ui_out *, instantiate a
cli_ui_out.
* cli-out.h (cli_ui_out_data): Remove, replace with class
cli_ui_out.
(class cli_ui_out): New class.
(cli_ui_out_impl): Remove.
(cli_out_data_ctor): Remove.
(cli_out_new): Change return type to cli_ui_out*.
(cli_out_set_stream): Remove.
* cli/cli-interp.c (struct cli_interp) <cli_uiout>: Change type
to cli_ui_out*.
(cli_interpreter_resume): Adapt.
(cli_interpreter_exec): Adapt.
* mi/mi-out.c (mi_ui_out_data, mi_out_data): Remove.
(mi_ui_out_impl): Remove.
(mi_table_begin): Replace with ...
(mi_ui_out::do_table_begin): ... this.
(mi_table_body): Replace with ...
(mi_ui_out::do_table_body): ... this.
(mi_table_end): Replace with ...
(mi_ui_out::do_table_end): ... this.
(mi_table_header): Replace with ...
(mi_ui_out::do_table_header): ... this.
(mi_begin): Replace with ...
(mi_ui_out::do_begin): ... this.
(mi_end): Replace with ...
(mi_ui_out::do_end): ... this.
(mi_field_int): Replace with ...
(mi_ui_out::do_field_int): ... this.
(mi_field_skip): Replace with ...
(mi_ui_out::do_field_skip): ... this.
(mi_field_string): Replace with ...
(mi_ui_out::do_field_string): ... this.
(mi_field_fmt): Replace with ...
(mi_ui_out::do_field_fmt): ... this.
(mi_spaces): Replace with ...
(mi_ui_out::do_spaces): ... this.
(mi_text): Replace with ...
(mi_ui_out::do_text): ... this.
(mi_message): Replace with ...
(mi_ui_out::do_message): ... this.
(mi_wrap_hint): Replace with ...
(mi_ui_out::do_wrap_hint): ... this.
(mi_flush): Replace with ...
(mi_ui_out::do_flush): ... this.
(mi_redirect): Replace with ...
(mi_ui_out::do_redirect):
(field_separator): Replace with ...
(mi_ui_out::field_separator):
(mi_open): Replace with ...
(mi_ui_out::open): ... this.
(mi_close): Replace with ...
(mi_ui_out::close): ... this.
(mi_out_rewind): Replace with ...
(mi_ui_out::rewind): ... this.
(mi_out_put): Replace with ...
(mi_ui_out::put): ... this.
(mi_version): Replace with ...
(mi_ui_out::version): ... this.
(mi_out_data_ctor): Replace with ...
(mi_ui_out::mi_ui_out): ... this.
(mi_out_data_dtor): Replace with ...
(mi_ui_out::~mi_ui_out): ... this.
(mi_out_new): Change return type to mi_ui_out*, instantiate
an mi_ui_out object.
(as_mi_ui_out): New function.
(mi_version): Update fallouts of struct ui_out to class ui_out
transition.
(mi_out_put): Likewise.
(mi_out_rewind): Likewise.
* mi/mi-out.h (mi_out_new): Change return type to mi_ui_out*.
* tui/tui-out.c (tui_ui_out_data, tui_out_data, tui_ui_out_impl):
Remove.
(tui_field_int): Replace with ...
(tui_ui_out::do_field_int): ... this.
(tui_field_string): Replace with ...
(tui_ui_out::do_field_string): ... this.
(tui_field_fmt): Replace with ...
(tui_ui_out::do_field_fmt): ... this.
(tui_text): Replace with ...
(tui_ui_out::do_text): ... this.
(tui_out_new): Change return type to tui_ui_out*, instantiate
tui_ui_out object.
(tui_ui_out::tui_ui_out): New.
* tui/tui-out.h: New file.
* tui/tui.h (tui_out_new): Move declaration to tui/tui-out.h.
* tui/tui-io.c: Include tui/tui-out.h.
(tui_old_uiout): Change type to cli_ui_out*.
(tui_setup_io): Use dynamic_cast.
* tui/tui-io.h (tui_old_uiout): Change type to cli_ui_out*.
* tui/tui-interp.c (tui_resume): Adapt.
* ada-lang.c (print_it_exception): Update fallouts of struct
ui_out to class ui_out transition.
(print_one_exception): Likewise.
(print_mention_exception): Likewise.
* ada-tasks.c (print_ada_task_info): Likewise.
(info_task): Likewise.
(task_command): Likewise.
* auto-load.c (print_script): Likewise.
(auto_load_info_scripts): Likewise.
(info_auto_load_cmd): Likewise.
* break-catch-sig.c (signal_catchpoint_print_one): Likewise.
* break-catch-syscall.c (print_it_catch_syscall): Likewise.
(print_one_catch_syscall): Likewise.
* break-catch-throw.c (print_it_exception_catchpoint): Likewise.
(print_one_exception_catchpoint): Likewise.
(print_one_detail_exception_catchpoint): Likewise.
(print_mention_exception_catchpoint): Likewise.
* breakpoint.c (maybe_print_thread_hit_breakpoint): Likewise.
(print_solib_event): Likewise.
(watchpoint_check): Likewise.
(wrap_indent_at_field): Likewise.
(print_breakpoint_location): Likewise.
(output_thread_groups): Likewise.
(print_one_breakpoint_location): Likewise.
(breakpoint_1): Likewise.
(default_collect_info): Likewise.
(watchpoints_info): Likewise.
(print_it_catch_fork): Likewise.
(print_one_catch_fork): Likewise.
(print_it_catch_vfork): Likewise.
(print_one_catch_vfork): Likewise.
(print_it_catch_solib): Likewise.
(print_one_catch_solib): Likewise.
(print_it_catch_exec): Likewise.
(print_one_catch_exec): Likewise.
(mention): Likewise.
(print_it_ranged_breakpoint): Likewise.
(print_one_ranged_breakpoint): Likewise.
(print_one_detail_ranged_breakpoint): Likewise.
(print_mention_ranged_breakpoint): Likewise.
(print_it_watchpoint): Likewise.
(print_mention_watchpoint): Likewise.
(print_it_masked_watchpoint): Likewise.
(print_one_detail_masked_watchpoint): Likewise.
(print_mention_masked_watchpoint): Likewise.
(bkpt_print_it): Likewise.
(tracepoint_print_one_detail): Likewise.
(tracepoint_print_mention): Likewise.
(update_static_tracepoint): Likewise.
(tracepoints_info): Likewise.
(save_breakpoints): Likewise.
* cli/cli-cmds.c (complete_command): Likewise.
* cli/cli-logging.c (set_logging_redirect): Likewise.
(pop_output_files): Likewise.
(handle_redirections): Likewise.
* cli/cli-script.c (print_command_lines): Likewise.
* cli/cli-setshow.c (do_show_command): Likewise.
(cmd_show_list): Likewise.
* cp-abi.c (list_cp_abis): Likewise.
(show_cp_abi_cmd): Likewise.
* darwin-nat-info.c (darwin_debug_regions_recurse): Likewise.
* disasm.c (gdb_pretty_print_insn): Likewise.
(do_mixed_source_and_assembly_deprecated): Likewise.
(do_mixed_source_and_assembly): Likewise.
* gdb_bfd.c (print_one_bfd): Likewise.
(maintenance_info_bfds): Likewise.
* guile/scm-breakpoint.c (gdbscm_breakpoint_commands): Likewise.
* guile/scm-ports.c (ioscm_with_output_to_port_worker): Likewise.
* i386-linux-tdep.c (i386_linux_handle_segmentation_fault): Likewise.
* i386-tdep.c (i386_mpx_print_bounds): Likewise.
* infcmd.c (run_command_1): Likewise.
(print_return_value_1): Likewise.
* inferior.c (print_selected_inferior): Likewise.
(print_inferior): Likewise.
* infrun.c (print_end_stepping_range_reason): Likewise.
(print_signal_exited_reason): Likewise.
(print_exited_reason): Likewise.
(print_signal_received_reason): Likewise.
(print_no_history_reason): Likewise.
* interps.c (interp_set): Likewise.
* linespec.c (decode_line_full): Likewise.
* linux-thread-db.c (info_auto_load_libthread_db): Likewise.
* mi/mi-cmd-env.c (mi_cmd_env_pwd): Likewise.
(mi_cmd_env_path): Likewise.
(mi_cmd_env_dir): Likewise.
(mi_cmd_inferior_tty_show): Likewise.
* mi/mi-cmd-file.c (mi_cmd_file_list_exec_source_file): Likewise.
(print_partial_file_name): Likewise.
(mi_cmd_file_list_exec_source_files): Likewise.
* mi/mi-cmd-info.c (mi_cmd_info_ada_exceptions): Likewise.
(mi_cmd_info_gdb_mi_command): Likewise.
* mi/mi-cmd-stack.c (mi_cmd_stack_info_depth): Likewise.
(mi_cmd_stack_list_args): Likewise.
(list_arg_or_local): Likewise.
* mi/mi-cmd-var.c (print_varobj): Likewise.
(mi_cmd_var_create): Likewise.
(mi_cmd_var_delete): Likewise.
(mi_cmd_var_set_format): Likewise.
(mi_cmd_var_show_format): Likewise.
(mi_cmd_var_info_num_children): Likewise.
(mi_cmd_var_list_children): Likewise.
(mi_cmd_var_info_type): Likewise.
(mi_cmd_var_info_path_expression): Likewise.
(mi_cmd_var_info_expression): Likewise.
(mi_cmd_var_show_attributes): Likewise.
(mi_cmd_var_evaluate_expression): Likewise.
(mi_cmd_var_assign): Likewise.
(varobj_update_one): Likewise.
* mi/mi-interp.c (as_mi_interp): Likewise.
(mi_on_normal_stop_1): Likewise.
(mi_tsv_modified): Likewise.
(mi_breakpoint_created): Likewise.
(mi_breakpoint_modified): Likewise.
(mi_solib_loaded): Likewise.
(mi_solib_unloaded): Likewise.
(mi_command_param_changed): Likewise.
(mi_memory_changed): Likewise.
(mi_user_selected_context_changed): Likewise.
* mi/mi-main.c (print_one_inferior): Likewise.
(output_cores): Likewise.
(list_available_thread_groups): Likewise.
(mi_cmd_data_list_register_names): Likewise.
(mi_cmd_data_list_changed_registers): Likewise.
(output_register): Likewise.
(mi_cmd_data_evaluate_expression): Likewise.
(mi_cmd_data_read_memory): Likewise.
(mi_cmd_data_read_memory_bytes): Likewise.
(mi_cmd_list_features): Likewise.
(mi_cmd_list_target_features): Likewise.
(mi_cmd_add_inferior): Likewise.
(mi_execute_command): Likewise.
(mi_load_progress): Likewise.
(print_variable_or_computed): Likewise.
(mi_cmd_trace_frame_collected): Likewise.
* mi/mi-symbol-cmds.c (mi_cmd_symbol_list_lines): Likewise.
* osdata.c (info_osdata_command): Likewise.
* probe.c (gen_ui_out_table_header_info): Likewise.
(print_ui_out_not_applicables): Likewise.
(print_ui_out_info): Likewise.
(info_probes_for_ops): Likewise.
(enable_probes_command): Likewise.
(disable_probes_command): Likewise.
* progspace.c (print_program_space): Likewise.
* python/py-breakpoint.c (bppy_get_commands): Likewise.
* python/py-framefilter.c (py_print_type): Likewise.
(py_print_value): Likewise.
(py_print_single_arg): Likewise.
(enumerate_args): Likewise.
(enumerate_locals): Likewise.
(py_print_args): Likewise.
(py_print_frame): Likewise.
* record-btrace.c (btrace_ui_out_decode_error): Likewise.
(btrace_call_history_insn_range): Likewise.
(btrace_call_history_src_line): Likewise.
(btrace_call_history): Likewise.
* remote.c (show_remote_cmd): Likewise.
* skip.c (skip_info): Likewise.
* solib.c (info_sharedlibrary_command): Likewise.
* source.c (print_source_lines_base): Likewise.
* spu-tdep.c (info_spu_event_command): Likewise.
(info_spu_signal_command): Likewise.
(info_spu_mailbox_list): Likewise.
(info_spu_dma_cmdlist): Likewise.
(info_spu_dma_command): Likewise.
(info_spu_proxydma_command): Likewise.
* stack.c (print_stack_frame): Likewise.
(print_frame_arg): Likewise.
(read_frame_arg): Likewise.
(print_frame_args): Likewise.
(print_frame_info): Likewise.
(print_frame): Likewise.
* symfile.c (load_progress): Likewise.
(generic_load): Likewise.
(print_transfer_performance): Likewise.
* thread.c (do_captured_list_thread_ids): Likewise.
(print_thread_info_1): Likewise.
(restore_selected_frame): Likewise.
(do_captured_thread_select): Likewise.
(print_selected_thread_frame): Likewise.
* top.c (execute_command_to_string): Likewise.
* tracepoint.c (tvariables_info_1): Likewise.
(trace_status_mi): Likewise.
(tfind_1): Likewise.
(print_one_static_tracepoint_marker): Likewise.
(info_static_tracepoint_markers_command): Likewise.
* utils.c (do_ui_out_redirect_pop): Likewise.
(fputs_maybe_filtered): Likewise.
737 lines
21 KiB
C
737 lines
21 KiB
C
/* Everything about syscall catchpoints, for GDB.
|
|
|
|
Copyright (C) 2009-2016 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 <ctype.h>
|
|
#include "breakpoint.h"
|
|
#include "gdbcmd.h"
|
|
#include "inferior.h"
|
|
#include "cli/cli-utils.h"
|
|
#include "annotate.h"
|
|
#include "mi/mi-common.h"
|
|
#include "valprint.h"
|
|
#include "arch-utils.h"
|
|
#include "observer.h"
|
|
#include "xml-syscall.h"
|
|
|
|
/* An instance of this type is used to represent a syscall catchpoint.
|
|
It includes a "struct breakpoint" as a kind of base class; users
|
|
downcast to "struct breakpoint *" when needed. A breakpoint is
|
|
really of this type iff its ops pointer points to
|
|
CATCH_SYSCALL_BREAKPOINT_OPS. */
|
|
|
|
struct syscall_catchpoint
|
|
{
|
|
/* The base class. */
|
|
struct breakpoint base;
|
|
|
|
/* Syscall numbers used for the 'catch syscall' feature. If no
|
|
syscall has been specified for filtering, its value is NULL.
|
|
Otherwise, it holds a list of all syscalls to be caught. The
|
|
list elements are allocated with xmalloc. */
|
|
VEC(int) *syscalls_to_be_caught;
|
|
};
|
|
|
|
/* Implement the "dtor" breakpoint_ops method for syscall
|
|
catchpoints. */
|
|
|
|
static void
|
|
dtor_catch_syscall (struct breakpoint *b)
|
|
{
|
|
struct syscall_catchpoint *c = (struct syscall_catchpoint *) b;
|
|
|
|
VEC_free (int, c->syscalls_to_be_caught);
|
|
|
|
base_breakpoint_ops.dtor (b);
|
|
}
|
|
|
|
static const struct inferior_data *catch_syscall_inferior_data = NULL;
|
|
|
|
struct catch_syscall_inferior_data
|
|
{
|
|
/* We keep a count of the number of times the user has requested a
|
|
particular syscall to be tracked, and pass this information to the
|
|
target. This lets capable targets implement filtering directly. */
|
|
|
|
/* Number of times that "any" syscall is requested. */
|
|
int any_syscall_count;
|
|
|
|
/* Count of each system call. */
|
|
VEC(int) *syscalls_counts;
|
|
|
|
/* This counts all syscall catch requests, so we can readily determine
|
|
if any catching is necessary. */
|
|
int total_syscalls_count;
|
|
};
|
|
|
|
static struct catch_syscall_inferior_data*
|
|
get_catch_syscall_inferior_data (struct inferior *inf)
|
|
{
|
|
struct catch_syscall_inferior_data *inf_data;
|
|
|
|
inf_data = ((struct catch_syscall_inferior_data *)
|
|
inferior_data (inf, catch_syscall_inferior_data));
|
|
if (inf_data == NULL)
|
|
{
|
|
inf_data = XCNEW (struct catch_syscall_inferior_data);
|
|
set_inferior_data (inf, catch_syscall_inferior_data, inf_data);
|
|
}
|
|
|
|
return inf_data;
|
|
}
|
|
|
|
static void
|
|
catch_syscall_inferior_data_cleanup (struct inferior *inf, void *arg)
|
|
{
|
|
xfree (arg);
|
|
}
|
|
|
|
|
|
/* Implement the "insert" breakpoint_ops method for syscall
|
|
catchpoints. */
|
|
|
|
static int
|
|
insert_catch_syscall (struct bp_location *bl)
|
|
{
|
|
struct syscall_catchpoint *c = (struct syscall_catchpoint *) bl->owner;
|
|
struct inferior *inf = current_inferior ();
|
|
struct catch_syscall_inferior_data *inf_data
|
|
= get_catch_syscall_inferior_data (inf);
|
|
|
|
++inf_data->total_syscalls_count;
|
|
if (!c->syscalls_to_be_caught)
|
|
++inf_data->any_syscall_count;
|
|
else
|
|
{
|
|
int i, iter;
|
|
|
|
for (i = 0;
|
|
VEC_iterate (int, c->syscalls_to_be_caught, i, iter);
|
|
i++)
|
|
{
|
|
int elem;
|
|
|
|
if (iter >= VEC_length (int, inf_data->syscalls_counts))
|
|
{
|
|
int old_size = VEC_length (int, inf_data->syscalls_counts);
|
|
uintptr_t vec_addr_offset
|
|
= old_size * ((uintptr_t) sizeof (int));
|
|
uintptr_t vec_addr;
|
|
VEC_safe_grow (int, inf_data->syscalls_counts, iter + 1);
|
|
vec_addr = ((uintptr_t) VEC_address (int,
|
|
inf_data->syscalls_counts)
|
|
+ vec_addr_offset);
|
|
memset ((void *) vec_addr, 0,
|
|
(iter + 1 - old_size) * sizeof (int));
|
|
}
|
|
elem = VEC_index (int, inf_data->syscalls_counts, iter);
|
|
VEC_replace (int, inf_data->syscalls_counts, iter, ++elem);
|
|
}
|
|
}
|
|
|
|
return target_set_syscall_catchpoint (ptid_get_pid (inferior_ptid),
|
|
inf_data->total_syscalls_count != 0,
|
|
inf_data->any_syscall_count,
|
|
VEC_length (int,
|
|
inf_data->syscalls_counts),
|
|
VEC_address (int,
|
|
inf_data->syscalls_counts));
|
|
}
|
|
|
|
/* Implement the "remove" breakpoint_ops method for syscall
|
|
catchpoints. */
|
|
|
|
static int
|
|
remove_catch_syscall (struct bp_location *bl, enum remove_bp_reason reason)
|
|
{
|
|
struct syscall_catchpoint *c = (struct syscall_catchpoint *) bl->owner;
|
|
struct inferior *inf = current_inferior ();
|
|
struct catch_syscall_inferior_data *inf_data
|
|
= get_catch_syscall_inferior_data (inf);
|
|
|
|
--inf_data->total_syscalls_count;
|
|
if (!c->syscalls_to_be_caught)
|
|
--inf_data->any_syscall_count;
|
|
else
|
|
{
|
|
int i, iter;
|
|
|
|
for (i = 0;
|
|
VEC_iterate (int, c->syscalls_to_be_caught, i, iter);
|
|
i++)
|
|
{
|
|
int elem;
|
|
if (iter >= VEC_length (int, inf_data->syscalls_counts))
|
|
/* Shouldn't happen. */
|
|
continue;
|
|
elem = VEC_index (int, inf_data->syscalls_counts, iter);
|
|
VEC_replace (int, inf_data->syscalls_counts, iter, --elem);
|
|
}
|
|
}
|
|
|
|
return target_set_syscall_catchpoint (ptid_get_pid (inferior_ptid),
|
|
inf_data->total_syscalls_count != 0,
|
|
inf_data->any_syscall_count,
|
|
VEC_length (int,
|
|
inf_data->syscalls_counts),
|
|
VEC_address (int,
|
|
inf_data->syscalls_counts));
|
|
}
|
|
|
|
/* Implement the "breakpoint_hit" breakpoint_ops method for syscall
|
|
catchpoints. */
|
|
|
|
static int
|
|
breakpoint_hit_catch_syscall (const struct bp_location *bl,
|
|
struct address_space *aspace, CORE_ADDR bp_addr,
|
|
const struct target_waitstatus *ws)
|
|
{
|
|
/* We must check if we are catching specific syscalls in this
|
|
breakpoint. If we are, then we must guarantee that the called
|
|
syscall is the same syscall we are catching. */
|
|
int syscall_number = 0;
|
|
const struct syscall_catchpoint *c
|
|
= (const struct syscall_catchpoint *) bl->owner;
|
|
|
|
if (ws->kind != TARGET_WAITKIND_SYSCALL_ENTRY
|
|
&& ws->kind != TARGET_WAITKIND_SYSCALL_RETURN)
|
|
return 0;
|
|
|
|
syscall_number = ws->value.syscall_number;
|
|
|
|
/* Now, checking if the syscall is the same. */
|
|
if (c->syscalls_to_be_caught)
|
|
{
|
|
int i, iter;
|
|
|
|
for (i = 0;
|
|
VEC_iterate (int, c->syscalls_to_be_caught, i, iter);
|
|
i++)
|
|
if (syscall_number == iter)
|
|
return 1;
|
|
|
|
return 0;
|
|
}
|
|
|
|
return 1;
|
|
}
|
|
|
|
/* Implement the "print_it" breakpoint_ops method for syscall
|
|
catchpoints. */
|
|
|
|
static enum print_stop_action
|
|
print_it_catch_syscall (bpstat bs)
|
|
{
|
|
struct ui_out *uiout = current_uiout;
|
|
struct breakpoint *b = bs->breakpoint_at;
|
|
/* These are needed because we want to know in which state a
|
|
syscall is. It can be in the TARGET_WAITKIND_SYSCALL_ENTRY
|
|
or TARGET_WAITKIND_SYSCALL_RETURN, and depending on it we
|
|
must print "called syscall" or "returned from syscall". */
|
|
ptid_t ptid;
|
|
struct target_waitstatus last;
|
|
struct syscall s;
|
|
struct gdbarch *gdbarch = bs->bp_location_at->gdbarch;
|
|
|
|
get_last_target_status (&ptid, &last);
|
|
|
|
get_syscall_by_number (gdbarch, last.value.syscall_number, &s);
|
|
|
|
annotate_catchpoint (b->number);
|
|
maybe_print_thread_hit_breakpoint (uiout);
|
|
|
|
if (b->disposition == disp_del)
|
|
uiout->text ("Temporary catchpoint ");
|
|
else
|
|
uiout->text ("Catchpoint ");
|
|
if (uiout->is_mi_like_p ())
|
|
{
|
|
uiout->field_string ("reason",
|
|
async_reason_lookup (last.kind == TARGET_WAITKIND_SYSCALL_ENTRY
|
|
? EXEC_ASYNC_SYSCALL_ENTRY
|
|
: EXEC_ASYNC_SYSCALL_RETURN));
|
|
uiout->field_string ("disp", bpdisp_text (b->disposition));
|
|
}
|
|
uiout->field_int ("bkptno", b->number);
|
|
|
|
if (last.kind == TARGET_WAITKIND_SYSCALL_ENTRY)
|
|
uiout->text (" (call to syscall ");
|
|
else
|
|
uiout->text (" (returned from syscall ");
|
|
|
|
if (s.name == NULL || uiout->is_mi_like_p ())
|
|
uiout->field_int ("syscall-number", last.value.syscall_number);
|
|
if (s.name != NULL)
|
|
uiout->field_string ("syscall-name", s.name);
|
|
|
|
uiout->text ("), ");
|
|
|
|
return PRINT_SRC_AND_LOC;
|
|
}
|
|
|
|
/* Implement the "print_one" breakpoint_ops method for syscall
|
|
catchpoints. */
|
|
|
|
static void
|
|
print_one_catch_syscall (struct breakpoint *b,
|
|
struct bp_location **last_loc)
|
|
{
|
|
struct syscall_catchpoint *c = (struct syscall_catchpoint *) b;
|
|
struct value_print_options opts;
|
|
struct ui_out *uiout = current_uiout;
|
|
struct gdbarch *gdbarch = b->loc->gdbarch;
|
|
|
|
get_user_print_options (&opts);
|
|
/* Field 4, the address, is omitted (which makes the columns not
|
|
line up too nicely with the headers, but the effect is relatively
|
|
readable). */
|
|
if (opts.addressprint)
|
|
uiout->field_skip ("addr");
|
|
annotate_field (5);
|
|
|
|
if (c->syscalls_to_be_caught
|
|
&& VEC_length (int, c->syscalls_to_be_caught) > 1)
|
|
uiout->text ("syscalls \"");
|
|
else
|
|
uiout->text ("syscall \"");
|
|
|
|
if (c->syscalls_to_be_caught)
|
|
{
|
|
int i, iter;
|
|
char *text = xstrprintf ("%s", "");
|
|
|
|
for (i = 0;
|
|
VEC_iterate (int, c->syscalls_to_be_caught, i, iter);
|
|
i++)
|
|
{
|
|
char *x = text;
|
|
struct syscall s;
|
|
get_syscall_by_number (gdbarch, iter, &s);
|
|
|
|
if (s.name != NULL)
|
|
text = xstrprintf ("%s%s, ", text, s.name);
|
|
else
|
|
text = xstrprintf ("%s%d, ", text, iter);
|
|
|
|
/* We have to xfree the last 'text' (now stored at 'x')
|
|
because xstrprintf dynamically allocates new space for it
|
|
on every call. */
|
|
xfree (x);
|
|
}
|
|
/* Remove the last comma. */
|
|
text[strlen (text) - 2] = '\0';
|
|
uiout->field_string ("what", text);
|
|
}
|
|
else
|
|
uiout->field_string ("what", "<any syscall>");
|
|
uiout->text ("\" ");
|
|
|
|
if (uiout->is_mi_like_p ())
|
|
uiout->field_string ("catch-type", "syscall");
|
|
}
|
|
|
|
/* Implement the "print_mention" breakpoint_ops method for syscall
|
|
catchpoints. */
|
|
|
|
static void
|
|
print_mention_catch_syscall (struct breakpoint *b)
|
|
{
|
|
struct syscall_catchpoint *c = (struct syscall_catchpoint *) b;
|
|
struct gdbarch *gdbarch = b->loc->gdbarch;
|
|
|
|
if (c->syscalls_to_be_caught)
|
|
{
|
|
int i, iter;
|
|
|
|
if (VEC_length (int, c->syscalls_to_be_caught) > 1)
|
|
printf_filtered (_("Catchpoint %d (syscalls"), b->number);
|
|
else
|
|
printf_filtered (_("Catchpoint %d (syscall"), b->number);
|
|
|
|
for (i = 0;
|
|
VEC_iterate (int, c->syscalls_to_be_caught, i, iter);
|
|
i++)
|
|
{
|
|
struct syscall s;
|
|
get_syscall_by_number (gdbarch, iter, &s);
|
|
|
|
if (s.name)
|
|
printf_filtered (" '%s' [%d]", s.name, s.number);
|
|
else
|
|
printf_filtered (" %d", s.number);
|
|
}
|
|
printf_filtered (")");
|
|
}
|
|
else
|
|
printf_filtered (_("Catchpoint %d (any syscall)"),
|
|
b->number);
|
|
}
|
|
|
|
/* Implement the "print_recreate" breakpoint_ops method for syscall
|
|
catchpoints. */
|
|
|
|
static void
|
|
print_recreate_catch_syscall (struct breakpoint *b, struct ui_file *fp)
|
|
{
|
|
struct syscall_catchpoint *c = (struct syscall_catchpoint *) b;
|
|
struct gdbarch *gdbarch = b->loc->gdbarch;
|
|
|
|
fprintf_unfiltered (fp, "catch syscall");
|
|
|
|
if (c->syscalls_to_be_caught)
|
|
{
|
|
int i, iter;
|
|
|
|
for (i = 0;
|
|
VEC_iterate (int, c->syscalls_to_be_caught, i, iter);
|
|
i++)
|
|
{
|
|
struct syscall s;
|
|
|
|
get_syscall_by_number (gdbarch, iter, &s);
|
|
if (s.name)
|
|
fprintf_unfiltered (fp, " %s", s.name);
|
|
else
|
|
fprintf_unfiltered (fp, " %d", s.number);
|
|
}
|
|
}
|
|
print_recreate_thread (b, fp);
|
|
}
|
|
|
|
/* The breakpoint_ops structure to be used in syscall catchpoints. */
|
|
|
|
static struct breakpoint_ops catch_syscall_breakpoint_ops;
|
|
|
|
/* Returns non-zero if 'b' is a syscall catchpoint. */
|
|
|
|
static int
|
|
syscall_catchpoint_p (struct breakpoint *b)
|
|
{
|
|
return (b->ops == &catch_syscall_breakpoint_ops);
|
|
}
|
|
|
|
static void
|
|
create_syscall_event_catchpoint (int tempflag, VEC(int) *filter,
|
|
const struct breakpoint_ops *ops)
|
|
{
|
|
struct syscall_catchpoint *c;
|
|
struct gdbarch *gdbarch = get_current_arch ();
|
|
|
|
c = new syscall_catchpoint ();
|
|
init_catchpoint (&c->base, gdbarch, tempflag, NULL, ops);
|
|
c->syscalls_to_be_caught = filter;
|
|
|
|
install_breakpoint (0, &c->base, 1);
|
|
}
|
|
|
|
/* Splits the argument using space as delimiter. Returns an xmalloc'd
|
|
filter list, or NULL if no filtering is required. */
|
|
static VEC(int) *
|
|
catch_syscall_split_args (char *arg)
|
|
{
|
|
VEC(int) *result = NULL;
|
|
struct cleanup *cleanup = make_cleanup (VEC_cleanup (int), &result);
|
|
struct gdbarch *gdbarch = target_gdbarch ();
|
|
|
|
while (*arg != '\0')
|
|
{
|
|
int i, syscall_number;
|
|
char *endptr;
|
|
char cur_name[128];
|
|
struct syscall s;
|
|
|
|
/* Skip whitespace. */
|
|
arg = skip_spaces (arg);
|
|
|
|
for (i = 0; i < 127 && arg[i] && !isspace (arg[i]); ++i)
|
|
cur_name[i] = arg[i];
|
|
cur_name[i] = '\0';
|
|
arg += i;
|
|
|
|
/* Check if the user provided a syscall name, group, or a number. */
|
|
syscall_number = (int) strtol (cur_name, &endptr, 0);
|
|
if (*endptr == '\0')
|
|
{
|
|
get_syscall_by_number (gdbarch, syscall_number, &s);
|
|
VEC_safe_push (int, result, s.number);
|
|
}
|
|
else if (startswith (cur_name, "g:")
|
|
|| startswith (cur_name, "group:"))
|
|
{
|
|
/* We have a syscall group. Let's expand it into a syscall
|
|
list before inserting. */
|
|
struct syscall *syscall_list;
|
|
const char *group_name;
|
|
|
|
/* Skip over "g:" and "group:" prefix strings. */
|
|
group_name = strchr (cur_name, ':') + 1;
|
|
|
|
syscall_list = get_syscalls_by_group (gdbarch, group_name);
|
|
|
|
if (syscall_list == NULL)
|
|
error (_("Unknown syscall group '%s'."), group_name);
|
|
|
|
for (i = 0; syscall_list[i].name != NULL; i++)
|
|
{
|
|
/* Insert each syscall that are part of the group. No
|
|
need to check if it is valid. */
|
|
VEC_safe_push (int, result, syscall_list[i].number);
|
|
}
|
|
|
|
xfree (syscall_list);
|
|
}
|
|
else
|
|
{
|
|
/* We have a name. Let's check if it's valid and convert it
|
|
to a number. */
|
|
get_syscall_by_name (gdbarch, cur_name, &s);
|
|
|
|
if (s.number == UNKNOWN_SYSCALL)
|
|
/* Here we have to issue an error instead of a warning,
|
|
because GDB cannot do anything useful if there's no
|
|
syscall number to be caught. */
|
|
error (_("Unknown syscall name '%s'."), cur_name);
|
|
|
|
/* Ok, it's valid. */
|
|
VEC_safe_push (int, result, s.number);
|
|
}
|
|
}
|
|
|
|
discard_cleanups (cleanup);
|
|
return result;
|
|
}
|
|
|
|
/* Implement the "catch syscall" command. */
|
|
|
|
static void
|
|
catch_syscall_command_1 (char *arg, int from_tty,
|
|
struct cmd_list_element *command)
|
|
{
|
|
int tempflag;
|
|
VEC(int) *filter;
|
|
struct syscall s;
|
|
struct gdbarch *gdbarch = get_current_arch ();
|
|
|
|
/* Checking if the feature if supported. */
|
|
if (gdbarch_get_syscall_number_p (gdbarch) == 0)
|
|
error (_("The feature 'catch syscall' is not supported on \
|
|
this architecture yet."));
|
|
|
|
tempflag = get_cmd_context (command) == CATCH_TEMPORARY;
|
|
|
|
arg = skip_spaces (arg);
|
|
|
|
/* We need to do this first "dummy" translation in order
|
|
to get the syscall XML file loaded or, most important,
|
|
to display a warning to the user if there's no XML file
|
|
for his/her architecture. */
|
|
get_syscall_by_number (gdbarch, 0, &s);
|
|
|
|
/* The allowed syntax is:
|
|
catch syscall
|
|
catch syscall <name | number> [<name | number> ... <name | number>]
|
|
|
|
Let's check if there's a syscall name. */
|
|
|
|
if (arg != NULL)
|
|
filter = catch_syscall_split_args (arg);
|
|
else
|
|
filter = NULL;
|
|
|
|
create_syscall_event_catchpoint (tempflag, filter,
|
|
&catch_syscall_breakpoint_ops);
|
|
}
|
|
|
|
|
|
/* Returns 0 if 'bp' is NOT a syscall catchpoint,
|
|
non-zero otherwise. */
|
|
static int
|
|
is_syscall_catchpoint_enabled (struct breakpoint *bp)
|
|
{
|
|
if (syscall_catchpoint_p (bp)
|
|
&& bp->enable_state != bp_disabled
|
|
&& bp->enable_state != bp_call_disabled)
|
|
return 1;
|
|
else
|
|
return 0;
|
|
}
|
|
|
|
int
|
|
catch_syscall_enabled (void)
|
|
{
|
|
struct catch_syscall_inferior_data *inf_data
|
|
= get_catch_syscall_inferior_data (current_inferior ());
|
|
|
|
return inf_data->total_syscalls_count != 0;
|
|
}
|
|
|
|
/* Helper function for catching_syscall_number. If B is a syscall
|
|
catchpoint for SYSCALL_NUMBER, return 1 (which will make
|
|
'breakpoint_find_if' return). Otherwise, return 0. */
|
|
|
|
static int
|
|
catching_syscall_number_1 (struct breakpoint *b,
|
|
void *data)
|
|
{
|
|
int syscall_number = (int) (uintptr_t) data;
|
|
|
|
if (is_syscall_catchpoint_enabled (b))
|
|
{
|
|
struct syscall_catchpoint *c = (struct syscall_catchpoint *) b;
|
|
|
|
if (c->syscalls_to_be_caught)
|
|
{
|
|
int i, iter;
|
|
for (i = 0;
|
|
VEC_iterate (int, c->syscalls_to_be_caught, i, iter);
|
|
i++)
|
|
if (syscall_number == iter)
|
|
return 1;
|
|
}
|
|
else
|
|
return 1;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
int
|
|
catching_syscall_number (int syscall_number)
|
|
{
|
|
struct breakpoint *b = breakpoint_find_if (catching_syscall_number_1,
|
|
(void *) (uintptr_t) syscall_number);
|
|
|
|
return b != NULL;
|
|
}
|
|
|
|
/* Complete syscall names. Used by "catch syscall". */
|
|
static VEC (char_ptr) *
|
|
catch_syscall_completer (struct cmd_list_element *cmd,
|
|
const char *text, const char *word)
|
|
{
|
|
struct gdbarch *gdbarch = get_current_arch ();
|
|
struct cleanup *cleanups = make_cleanup (null_cleanup, NULL);
|
|
VEC (char_ptr) *group_retlist = NULL;
|
|
VEC (char_ptr) *syscall_retlist = NULL;
|
|
VEC (char_ptr) *retlist = NULL;
|
|
const char **group_list = NULL;
|
|
const char **syscall_list = NULL;
|
|
const char *prefix;
|
|
int i;
|
|
|
|
/* Completion considers ':' to be a word separator, so we use this to
|
|
verify whether the previous word was a group prefix. If so, we
|
|
build the completion list using group names only. */
|
|
for (prefix = word; prefix != text && prefix[-1] != ' '; prefix--)
|
|
;
|
|
|
|
if (startswith (prefix, "g:") || startswith (prefix, "group:"))
|
|
{
|
|
/* Perform completion inside 'group:' namespace only. */
|
|
group_list = get_syscall_group_names (gdbarch);
|
|
retlist = (group_list == NULL
|
|
? NULL : complete_on_enum (group_list, word, word));
|
|
}
|
|
else
|
|
{
|
|
/* Complete with both, syscall names and groups. */
|
|
syscall_list = get_syscall_names (gdbarch);
|
|
group_list = get_syscall_group_names (gdbarch);
|
|
|
|
/* Append "group:" prefix to syscall groups. */
|
|
for (i = 0; group_list[i] != NULL; i++)
|
|
{
|
|
char *prefixed_group = xstrprintf ("group:%s", group_list[i]);
|
|
|
|
group_list[i] = prefixed_group;
|
|
make_cleanup (xfree, prefixed_group);
|
|
}
|
|
|
|
syscall_retlist = ((syscall_list == NULL)
|
|
? NULL : complete_on_enum (syscall_list, word, word));
|
|
group_retlist = ((group_list == NULL)
|
|
? NULL : complete_on_enum (group_list, word, word));
|
|
|
|
retlist = VEC_merge (char_ptr, syscall_retlist, group_retlist);
|
|
}
|
|
|
|
VEC_free (char_ptr, syscall_retlist);
|
|
VEC_free (char_ptr, group_retlist);
|
|
xfree (syscall_list);
|
|
xfree (group_list);
|
|
do_cleanups (cleanups);
|
|
|
|
return retlist;
|
|
}
|
|
|
|
static void
|
|
clear_syscall_counts (struct inferior *inf)
|
|
{
|
|
struct catch_syscall_inferior_data *inf_data
|
|
= get_catch_syscall_inferior_data (inf);
|
|
|
|
inf_data->total_syscalls_count = 0;
|
|
inf_data->any_syscall_count = 0;
|
|
VEC_free (int, inf_data->syscalls_counts);
|
|
}
|
|
|
|
static void
|
|
initialize_syscall_catchpoint_ops (void)
|
|
{
|
|
struct breakpoint_ops *ops;
|
|
|
|
initialize_breakpoint_ops ();
|
|
|
|
/* Syscall catchpoints. */
|
|
ops = &catch_syscall_breakpoint_ops;
|
|
*ops = base_breakpoint_ops;
|
|
ops->dtor = dtor_catch_syscall;
|
|
ops->insert_location = insert_catch_syscall;
|
|
ops->remove_location = remove_catch_syscall;
|
|
ops->breakpoint_hit = breakpoint_hit_catch_syscall;
|
|
ops->print_it = print_it_catch_syscall;
|
|
ops->print_one = print_one_catch_syscall;
|
|
ops->print_mention = print_mention_catch_syscall;
|
|
ops->print_recreate = print_recreate_catch_syscall;
|
|
}
|
|
|
|
initialize_file_ftype _initialize_break_catch_syscall;
|
|
|
|
void
|
|
_initialize_break_catch_syscall (void)
|
|
{
|
|
initialize_syscall_catchpoint_ops ();
|
|
|
|
observer_attach_inferior_exit (clear_syscall_counts);
|
|
catch_syscall_inferior_data
|
|
= register_inferior_data_with_cleanup (NULL,
|
|
catch_syscall_inferior_data_cleanup);
|
|
|
|
add_catch_command ("syscall", _("\
|
|
Catch system calls by their names, groups and/or numbers.\n\
|
|
Arguments say which system calls to catch. If no arguments are given,\n\
|
|
every system call will be caught. Arguments, if given, should be one\n\
|
|
or more system call names (if your system supports that), system call\n\
|
|
groups or system call numbers."),
|
|
catch_syscall_command_1,
|
|
catch_syscall_completer,
|
|
CATCH_PERMANENT,
|
|
CATCH_TEMPORARY);
|
|
}
|