mirror of
https://sourceware.org/git/binutils-gdb.git
synced 2024-12-13 12:23:33 +08:00
224c3ddb89
Most allocation functions (if not all) return a void* pointing to the allocated memory. In C++, we need to add an explicit cast when assigning the result to a pointer to another type (which is the case more often than not). The content of this patch is taken from Pedro's branch, from commit "(mostly) auto-generated patch to insert casts needed for C++". I validated that the changes make sense and manually reflowed the code to make it respect the coding style. I also found multiple places where I could use XNEW/XNEWVEC/XRESIZEVEC/etc. Thanks a lot to whoever did that automated script to insert casts, doing it completely by hand would have taken a ridiculous amount of time. Only files built on x86 with --enable-targets=all are modified. This means that all other -nat.c files are untouched and will have to be dealt with later by using appropiate compilers. Or maybe we can try to build them with a regular g++ just to know where to add casts, I don't know. I built-tested this with --enable-targets=all and reg-tested. Here's the changelog entry, which was not too bad to make despite the size, thanks to David Malcom's script. I fixed some bits by hand, but there might be some wrong parts left (hopefully not). gdb/ChangeLog: * aarch64-linux-tdep.c (aarch64_stap_parse_special_token): Add cast to allocation result assignment. * ada-exp.y (write_object_renaming): Likewise. (write_ambiguous_var): Likewise. (ada_nget_field_index): Likewise. (write_var_or_type): Likewise. * ada-lang.c (ada_decode_symbol): Likewise. (ada_value_assign): Likewise. (value_pointer): Likewise. (cache_symbol): Likewise. (add_nonlocal_symbols): Likewise. (ada_name_for_lookup): Likewise. (symbol_completion_add): Likewise. (ada_to_fixed_type_1): Likewise. (ada_get_next_arg): Likewise. (defns_collected): Likewise. * ada-lex.l (processId): Likewise. (processString): Likewise. * ada-tasks.c (read_known_tasks_array): Likewise. (read_known_tasks_list): Likewise. * ada-typeprint.c (decoded_type_name): Likewise. * addrmap.c (addrmap_mutable_create_fixed): Likewise. * amd64-tdep.c (amd64_push_arguments): Likewise. (amd64_displaced_step_copy_insn): Likewise. (amd64_classify_insn_at): Likewise. (amd64_relocate_instruction): Likewise. * amd64obsd-tdep.c (amd64obsd_sigtramp_p): Likewise. * arch-utils.c (simple_displaced_step_copy_insn): Likewise. (initialize_current_architecture): Likewise. * arm-linux-tdep.c (arm_stap_parse_special_token): Likewise. * arm-symbian-tdep.c (arm_symbian_osabi_sniffer): Likewise. * arm-tdep.c (arm_exidx_new_objfile): Likewise. (arm_push_dummy_call): Likewise. (extend_buffer_earlier): Likewise. (arm_adjust_breakpoint_address): Likewise. (arm_skip_stub): Likewise. * auto-load.c (filename_is_in_pattern): Likewise. (maybe_add_script_file): Likewise. (maybe_add_script_text): Likewise. (auto_load_objfile_script_1): Likewise. * auxv.c (ld_so_xfer_auxv): Likewise. * ax-general.c (new_agent_expr): Likewise. (grow_expr): Likewise. (ax_reg_mask): Likewise. * bcache.c (bcache_full): Likewise. * breakpoint.c (program_breakpoint_here_p): Likewise. * btrace.c (parse_xml_raw): Likewise. * build-id.c (build_id_to_debug_bfd): Likewise. * buildsym.c (end_symtab_with_blockvector): Likewise. * c-exp.y (string_exp): Likewise. (qualified_name): Likewise. (write_destructor_name): Likewise. (operator_stoken): Likewise. (parse_number): Likewise. (scan_macro_expansion): Likewise. (yylex): Likewise. (c_print_token): Likewise. * c-lang.c (c_get_string): Likewise. (emit_numeric_character): Likewise. * charset.c (wchar_iterate): Likewise. * cli/cli-cmds.c (complete_command): Likewise. (make_command): Likewise. * cli/cli-dump.c (restore_section_callback): Likewise. (restore_binary_file): Likewise. * cli/cli-interp.c (cli_interpreter_exec): Likewise. * cli/cli-script.c (execute_control_command): Likewise. * cli/cli-setshow.c (do_set_command): Likewise. * coff-pe-read.c (add_pe_forwarded_sym): Likewise. (read_pe_exported_syms): Likewise. * coffread.c (coff_read_struct_type): Likewise. (coff_read_enum_type): Likewise. * common/btrace-common.c (btrace_data_append): Likewise. * common/buffer.c (buffer_grow): Likewise. * common/filestuff.c (gdb_fopen_cloexec): Likewise. * common/format.c (parse_format_string): Likewise. * common/gdb_vecs.c (delim_string_to_char_ptr_vec_append): Likewise. * common/xml-utils.c (xml_escape_text): Likewise. * compile/compile-object-load.c (copy_sections): Likewise. (compile_object_load): Likewise. * compile/compile-object-run.c (compile_object_run): Likewise. * completer.c (filename_completer): Likewise. * corefile.c (read_memory_typed_address): Likewise. (write_memory_unsigned_integer): Likewise. (write_memory_signed_integer): Likewise. (complete_set_gnutarget): Likewise. * corelow.c (get_core_register_section): Likewise. * cp-name-parser.y (d_grab): Likewise. (allocate_info): Likewise. (cp_new_demangle_parse_info): Likewise. * cp-namespace.c (cp_scan_for_anonymous_namespaces): Likewise. (cp_lookup_symbol_in_namespace): Likewise. (lookup_namespace_scope): Likewise. (find_symbol_in_baseclass): Likewise. (cp_lookup_nested_symbol): Likewise. (cp_lookup_transparent_type_loop): Likewise. * cp-support.c (copy_string_to_obstack): Likewise. (make_symbol_overload_list): Likewise. (make_symbol_overload_list_namespace): Likewise. (make_symbol_overload_list_adl_namespace): Likewise. (first_component_command): Likewise. * cp-valprint.c (cp_print_value): Likewise. * ctf.c (ctf_xfer_partial): Likewise. * d-exp.y (StringExp): Likewise. * d-namespace.c (d_lookup_symbol_in_module): Likewise. (lookup_module_scope): Likewise. (find_symbol_in_baseclass): Likewise. (d_lookup_nested_symbol): Likewise. * dbxread.c (find_stab_function_addr): Likewise. (read_dbx_symtab): Likewise. (dbx_end_psymtab): Likewise. (cp_set_block_scope): Likewise. * dcache.c (dcache_alloc): Likewise. * demangle.c (_initialize_demangler): Likewise. * dicos-tdep.c (dicos_load_module_p): Likewise. * dictionary.c (dict_create_hashed_expandable): Likewise. (dict_create_linear_expandable): Likewise. (expand_hashtable): Likewise. (add_symbol_linear_expandable): Likewise. * dwarf2-frame.c (add_cie): Likewise. (add_fde): Likewise. (dwarf2_build_frame_info): Likewise. * dwarf2expr.c (dwarf_expr_grow_stack): Likewise. (dwarf_expr_fetch_address): Likewise. (add_piece): Likewise. (execute_stack_op): Likewise. * dwarf2loc.c (chain_candidate): Likewise. (dwarf_entry_parameter_to_value): Likewise. (read_pieced_value): Likewise. (write_pieced_value): Likewise. * dwarf2read.c (dwarf2_read_section): Likewise. (add_type_unit): Likewise. (read_comp_units_from_section): Likewise. (fixup_go_packaging): Likewise. (dwarf2_compute_name): Likewise. (dwarf2_physname): Likewise. (create_dwo_unit_in_dwp_v1): Likewise. (create_dwo_unit_in_dwp_v2): Likewise. (read_func_scope): Likewise. (read_call_site_scope): Likewise. (dwarf2_attach_fields_to_type): Likewise. (process_structure_scope): Likewise. (mark_common_block_symbol_computed): Likewise. (read_common_block): Likewise. (abbrev_table_read_table): Likewise. (guess_partial_die_structure_name): Likewise. (fixup_partial_die): Likewise. (add_file_name): Likewise. (dwarf2_const_value_data): Likewise. (dwarf2_const_value_attr): Likewise. (build_error_marker_type): Likewise. (guess_full_die_structure_name): Likewise. (anonymous_struct_prefix): Likewise. (typename_concat): Likewise. (dwarf2_canonicalize_name): Likewise. (dwarf2_name): Likewise. (write_constant_as_bytes): Likewise. (dwarf2_fetch_constant_bytes): Likewise. (copy_string): Likewise. (parse_macro_definition): Likewise. * elfread.c (elf_symfile_segments): Likewise. (elf_rel_plt_read): Likewise. (elf_gnu_ifunc_resolve_by_cache): Likewise. (elf_gnu_ifunc_resolve_by_got): Likewise. (elf_read_minimal_symbols): Likewise. (elf_gnu_ifunc_record_cache): Likewise. * event-top.c (top_level_prompt): Likewise. (command_line_handler): Likewise. * exec.c (resize_section_table): Likewise. * expprint.c (print_subexp_standard): Likewise. * fbsd-tdep.c (fbsd_collect_regset_section_cb): Likewise. * findcmd.c (parse_find_args): Likewise. * findvar.c (address_from_register): Likewise. * frame.c (get_prev_frame_always): Likewise. * gdb_bfd.c (gdb_bfd_ref): Likewise. (get_section_descriptor): Likewise. * gdb_obstack.c (obconcat): Likewise. (obstack_strdup): Likewise. * gdbtypes.c (lookup_function_type_with_arguments): Likewise. (create_set_type): Likewise. (lookup_unsigned_typename): Likewise. (lookup_signed_typename): Likewise. (resolve_dynamic_union): Likewise. (resolve_dynamic_struct): Likewise. (add_dyn_prop): Likewise. (copy_dynamic_prop_list): Likewise. (arch_flags_type): Likewise. (append_composite_type_field_raw): Likewise. * gdbtypes.h (INIT_FUNC_SPECIFIC): Likewise. * gnu-v3-abi.c (gnuv3_rtti_type): Likewise. * go-exp.y (string_exp): Likewise. * go-lang.c (go_demangle): Likewise. * guile/guile.c (compute_scheme_string): Likewise. * guile/scm-cmd.c (gdbscm_parse_command_name): Likewise. (gdbscm_canonicalize_command_name): Likewise. * guile/scm-ports.c (ioscm_init_stdio_buffers): Likewise. (ioscm_init_memory_port): Likewise. (ioscm_reinit_memory_port): Likewise. * guile/scm-utils.c (gdbscm_gc_xstrdup): Likewise. (gdbscm_gc_dup_argv): Likewise. * h8300-tdep.c (h8300_push_dummy_call): Likewise. * hppa-tdep.c (internalize_unwinds): Likewise. (read_unwind_info): Likewise. * i386-cygwin-tdep.c (core_process_module_section): Likewise. (windows_core_xfer_shared_libraries): Likewise. * i386-tdep.c (i386_displaced_step_copy_insn): Likewise. (i386_stap_parse_special_token_triplet): Likewise. (i386_stap_parse_special_token_three_arg_disp): Likewise. * i386obsd-tdep.c (i386obsd_sigtramp_p): Likewise. * inf-child.c (inf_child_fileio_readlink): Likewise. * inf-ptrace.c (inf_ptrace_fetch_register): Likewise. (inf_ptrace_store_register): Likewise. * infrun.c (follow_exec): Likewise. (displaced_step_prepare_throw): Likewise. (save_stop_context): Likewise. (save_infcall_suspend_state): Likewise. * jit.c (jit_read_descriptor): Likewise. (jit_read_code_entry): Likewise. (jit_symtab_line_mapping_add_impl): Likewise. (finalize_symtab): Likewise. (jit_unwind_reg_get_impl): Likewise. * jv-exp.y (QualifiedName): Likewise. * jv-lang.c (get_java_utf8_name): Likewise. (type_from_class): Likewise. (java_demangle_type_signature): Likewise. (java_class_name_from_physname): Likewise. * jv-typeprint.c (java_type_print_base): Likewise. * jv-valprint.c (java_value_print): Likewise. * language.c (add_language): Likewise. * linespec.c (add_sal_to_sals_basic): Likewise. (add_sal_to_sals): Likewise. (decode_objc): Likewise. (find_linespec_symbols): Likewise. * linux-fork.c (fork_save_infrun_state): Likewise. * linux-nat.c (linux_nat_detach): Likewise. (linux_nat_fileio_readlink): Likewise. * linux-record.c (record_linux_sockaddr): Likewise. (record_linux_msghdr): Likewise. (Do): Likewise. * linux-tdep.c (linux_core_info_proc_mappings): Likewise. (linux_collect_regset_section_cb): Likewise. (linux_get_siginfo_data): Likewise. * linux-thread-db.c (try_thread_db_load_from_pdir_1): Likewise. (try_thread_db_load_from_dir): Likewise. (thread_db_load_search): Likewise. (info_auto_load_libthread_db): Likewise. * m32c-tdep.c (m32c_m16c_address_to_pointer): Likewise. (m32c_m16c_pointer_to_address): Likewise. * m68hc11-tdep.c (m68hc11_pseudo_register_write): Likewise. * m68k-tdep.c (m68k_get_longjmp_target): Likewise. * machoread.c (macho_check_dsym): Likewise. * macroexp.c (resize_buffer): Likewise. (gather_arguments): Likewise. (maybe_expand): Likewise. * macrotab.c (new_macro_key): Likewise. (new_source_file): Likewise. (new_macro_definition): Likewise. * mdebugread.c (parse_symbol): Likewise. (parse_type): Likewise. (parse_partial_symbols): Likewise. (psymtab_to_symtab_1): Likewise. * mem-break.c (default_memory_insert_breakpoint): Likewise. * mi/mi-cmd-break.c (mi_argv_to_format): Likewise. * mi/mi-main.c (mi_cmd_data_read_memory): Likewise. (mi_cmd_data_read_memory_bytes): Likewise. (mi_cmd_data_write_memory_bytes): Likewise. (mi_cmd_trace_frame_collected): Likewise. * mi/mi-parse.c (mi_parse_argv): Likewise. (mi_parse): Likewise. * minidebug.c (lzma_open): Likewise. (lzma_pread): Likewise. * mips-tdep.c (mips_read_fp_register_single): Likewise. (mips_print_fp_register): Likewise. * mipsnbsd-tdep.c (mipsnbsd_get_longjmp_target): Likewise. * mipsread.c (read_alphacoff_dynamic_symtab): Likewise. * mt-tdep.c (mt_register_name): Likewise. (mt_registers_info): Likewise. (mt_push_dummy_call): Likewise. * namespace.c (add_using_directive): Likewise. * nat/linux-btrace.c (perf_event_read): Likewise. (linux_enable_bts): Likewise. * nat/linux-osdata.c (linux_common_core_of_thread): Likewise. * nat/linux-ptrace.c (linux_ptrace_test_ret_to_nx): Likewise. * nto-tdep.c (nto_find_and_open_solib): Likewise. (nto_parse_redirection): Likewise. * objc-lang.c (objc_demangle): Likewise. (find_methods): Likewise. * objfiles.c (get_objfile_bfd_data): Likewise. (set_objfile_main_name): Likewise. (allocate_objfile): Likewise. (objfile_relocate): Likewise. (update_section_map): Likewise. * osabi.c (generic_elf_osabi_sniff_abi_tag_sections): Likewise. * p-exp.y (exp): Likewise. (yylex): Likewise. * p-valprint.c (pascal_object_print_value): Likewise. * parse.c (initialize_expout): Likewise. (mark_completion_tag): Likewise. (copy_name): Likewise. (parse_float): Likewise. (type_stack_reserve): Likewise. * ppc-linux-tdep.c (ppc_stap_parse_special_token): Likewise. (ppu2spu_prev_register): Likewise. * ppc-ravenscar-thread.c (supply_register_at_address): Likewise. * printcmd.c (printf_wide_c_string): Likewise. (printf_pointer): Likewise. * probe.c (parse_probes): Likewise. * python/py-cmd.c (gdbpy_parse_command_name): Likewise. (cmdpy_init): Likewise. * python/py-gdb-readline.c (gdbpy_readline_wrapper): Likewise. * python/py-symtab.c (set_sal): Likewise. * python/py-unwind.c (pyuw_sniffer): Likewise. * python/python.c (python_interactive_command): Likewise. (compute_python_string): Likewise. * ravenscar-thread.c (get_running_thread_id): Likewise. * record-full.c (record_full_exec_insn): Likewise. (record_full_core_open_1): Likewise. * regcache.c (regcache_raw_read_signed): Likewise. (regcache_raw_read_unsigned): Likewise. (regcache_cooked_read_signed): Likewise. (regcache_cooked_read_unsigned): Likewise. * remote-fileio.c (remote_fileio_func_open): Likewise. (remote_fileio_func_rename): Likewise. (remote_fileio_func_unlink): Likewise. (remote_fileio_func_stat): Likewise. (remote_fileio_func_system): Likewise. * remote-mips.c (mips_xfer_memory): Likewise. (mips_load_srec): Likewise. (pmon_end_download): Likewise. * remote.c (new_remote_state): Likewise. (map_regcache_remote_table): Likewise. (remote_register_number_and_offset): Likewise. (init_remote_state): Likewise. (get_memory_packet_size): Likewise. (remote_pass_signals): Likewise. (remote_program_signals): Likewise. (remote_start_remote): Likewise. (remote_check_symbols): Likewise. (remote_query_supported): Likewise. (extended_remote_attach): Likewise. (process_g_packet): Likewise. (store_registers_using_G): Likewise. (putpkt_binary): Likewise. (read_frame): Likewise. (compare_sections_command): Likewise. (remote_hostio_pread): Likewise. (remote_hostio_readlink): Likewise. (remote_file_put): Likewise. (remote_file_get): Likewise. (remote_pid_to_exec_file): Likewise. (_initialize_remote): Likewise. * rs6000-aix-tdep.c (rs6000_aix_ld_info_to_xml): Likewise. (rs6000_aix_core_xfer_shared_libraries_aix): Likewise. * rs6000-tdep.c (ppc_displaced_step_copy_insn): Likewise. (bfd_uses_spe_extensions): Likewise. * s390-linux-tdep.c (s390_displaced_step_copy_insn): Likewise. * score-tdep.c (score7_malloc_and_get_memblock): Likewise. * solib-dsbt.c (decode_loadmap): Likewise. (fetch_loadmap): Likewise. (scan_dyntag): Likewise. (enable_break): Likewise. (dsbt_relocate_main_executable): Likewise. * solib-frv.c (fetch_loadmap): Likewise. (enable_break2): Likewise. (frv_relocate_main_executable): Likewise. * solib-spu.c (spu_relocate_main_executable): Likewise. (spu_bfd_open): Likewise. * solib-svr4.c (lm_info_read): Likewise. (read_program_header): Likewise. (find_program_interpreter): Likewise. (scan_dyntag): Likewise. (elf_locate_base): Likewise. (open_symbol_file_object): Likewise. (read_program_headers_from_bfd): Likewise. (svr4_relocate_main_executable): Likewise. * solib-target.c (solib_target_relocate_section_addresses): Likewise. * solib.c (solib_find_1): Likewise. (exec_file_find): Likewise. (solib_find): Likewise. * source.c (openp): Likewise. (print_source_lines_base): Likewise. (forward_search_command): Likewise. * sparc-ravenscar-thread.c (supply_register_at_address): Likewise. * spu-tdep.c (spu2ppu_prev_register): Likewise. (spu_get_overlay_table): Likewise. * stabsread.c (patch_block_stabs): Likewise. (define_symbol): Likewise. (again:): Likewise. (read_member_functions): Likewise. (read_one_struct_field): Likewise. (read_enum_type): Likewise. (common_block_start): Likewise. * stack.c (read_frame_arg): Likewise. (backtrace_command): Likewise. * stap-probe.c (stap_parse_register_operand): Likewise. * symfile.c (syms_from_objfile_1): Likewise. (find_separate_debug_file): Likewise. (load_command): Likewise. (load_progress): Likewise. (load_section_callback): Likewise. (reread_symbols): Likewise. (add_filename_language): Likewise. (allocate_compunit_symtab): Likewise. (read_target_long_array): Likewise. (simple_read_overlay_table): Likewise. * symtab.c (symbol_set_names): Likewise. (resize_symbol_cache): Likewise. (rbreak_command): Likewise. (completion_list_add_name): Likewise. (completion_list_objc_symbol): Likewise. (add_filename_to_list): Likewise. * target-descriptions.c (maint_print_c_tdesc_cmd): Likewise. * target-memory.c (target_write_memory_blocks): Likewise. * target.c (target_read_string): Likewise. (read_whatever_is_readable): Likewise. (target_read_alloc_1): Likewise. (simple_search_memory): Likewise. (target_fileio_read_alloc_1): Likewise. * tilegx-tdep.c (tilegx_push_dummy_call): Likewise. * top.c (command_line_input): Likewise. * tracefile-tfile.c (tfile_fetch_registers): Likewise. * tracefile.c (tracefile_fetch_registers): Likewise. * tracepoint.c (add_memrange): Likewise. (init_collection_list): Likewise. (add_aexpr): Likewise. (trace_dump_actions): Likewise. (parse_trace_status): Likewise. (parse_tracepoint_definition): Likewise. (parse_tsv_definition): Likewise. (parse_static_tracepoint_marker_definition): Likewise. * tui/tui-file.c (tui_sfileopen): Likewise. (tui_file_adjust_strbuf): Likewise. * tui/tui-io.c (tui_expand_tabs): Likewise. * tui/tui-source.c (tui_set_source_content): Likewise. * typeprint.c (find_global_typedef): Likewise. * ui-file.c (do_ui_file_xstrdup): Likewise. (ui_file_obsavestring): Likewise. (mem_file_write): Likewise. * utils.c (make_hex_string): Likewise. (get_regcomp_error): Likewise. (puts_filtered_tabular): Likewise. (gdb_realpath_keepfile): Likewise. (ldirname): Likewise. (gdb_bfd_errmsg): Likewise. (substitute_path_component): Likewise. * valops.c (search_struct_method): Likewise. (find_oload_champ_namespace_loop): Likewise. * valprint.c (print_decimal_chars): Likewise. (read_string): Likewise. (generic_emit_char): Likewise. * varobj.c (varobj_delete): Likewise. (varobj_value_get_print_value): Likewise. * vaxobsd-tdep.c (vaxobsd_sigtramp_sniffer): Likewise. * windows-tdep.c (display_one_tib): Likewise. * xcoffread.c (read_xcoff_symtab): Likewise. (process_xcoff_symbol): Likewise. (swap_sym): Likewise. (scan_xcoff_symtab): Likewise. (xcoff_initial_scan): Likewise. * xml-support.c (gdb_xml_end_element): Likewise. (xml_process_xincludes): Likewise. (xml_fetch_content_from_file): Likewise. * xml-syscall.c (xml_list_of_syscalls): Likewise. * xstormy16-tdep.c (xstormy16_push_dummy_call): Likewise. gdb/gdbserver/ChangeLog: * ax.c (gdb_parse_agent_expr): Add cast to allocation result assignment. (gdb_unparse_agent_expr): Likewise. * hostio.c (require_data): Likewise. (handle_pread): Likewise. * linux-low.c (disable_regset): Likewise. (fetch_register): Likewise. (store_register): Likewise. (get_dynamic): Likewise. (linux_qxfer_libraries_svr4): Likewise. * mem-break.c (delete_fast_tracepoint_jump): Likewise. (set_fast_tracepoint_jump): Likewise. (uninsert_fast_tracepoint_jumps_at): Likewise. (reinsert_fast_tracepoint_jumps_at): Likewise. (validate_inserted_breakpoint): Likewise. (clone_agent_expr): Likewise. * regcache.c (init_register_cache): Likewise. * remote-utils.c (putpkt_binary_1): Likewise. (decode_M_packet): Likewise. (decode_X_packet): Likewise. (look_up_one_symbol): Likewise. (relocate_instruction): Likewise. (monitor_output): Likewise. * server.c (handle_search_memory): Likewise. (handle_qxfer_exec_file): Likewise. (handle_qxfer_libraries): Likewise. (handle_qxfer): Likewise. (handle_query): Likewise. (handle_v_cont): Likewise. (handle_v_run): Likewise. (captured_main): Likewise. * target.c (write_inferior_memory): Likewise. * thread-db.c (try_thread_db_load_from_dir): Likewise. * tracepoint.c (init_trace_buffer): Likewise. (add_tracepoint_action): Likewise. (add_traceframe): Likewise. (add_traceframe_block): Likewise. (cmd_qtdpsrc): Likewise. (cmd_qtdv): Likewise. (cmd_qtstatus): Likewise. (response_source): Likewise. (response_tsv): Likewise. (cmd_qtnotes): Likewise. (gdb_collect): Likewise. (initialize_tracepoint): Likewise.
1993 lines
55 KiB
C
1993 lines
55 KiB
C
/* GDB CLI commands.
|
||
|
||
Copyright (C) 2000-2015 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 "arch-utils.h"
|
||
#include "dyn-string.h"
|
||
#include "readline/readline.h"
|
||
#include "readline/tilde.h"
|
||
#include "completer.h"
|
||
#include "target.h" /* For baud_rate, remote_debug and remote_timeout. */
|
||
#include "gdb_wait.h" /* For shell escape implementation. */
|
||
#include "gdb_regex.h" /* Used by apropos_command. */
|
||
#include "gdb_vfork.h"
|
||
#include "linespec.h"
|
||
#include "expression.h"
|
||
#include "frame.h"
|
||
#include "value.h"
|
||
#include "language.h"
|
||
#include "filenames.h" /* For DOSish file names. */
|
||
#include "objfiles.h"
|
||
#include "source.h"
|
||
#include "disasm.h"
|
||
#include "tracepoint.h"
|
||
#include "filestuff.h"
|
||
#include "location.h"
|
||
|
||
#include "ui-out.h"
|
||
|
||
#include "top.h"
|
||
#include "cli/cli-decode.h"
|
||
#include "cli/cli-script.h"
|
||
#include "cli/cli-setshow.h"
|
||
#include "cli/cli-cmds.h"
|
||
#include "cli/cli-utils.h"
|
||
|
||
#include "extension.h"
|
||
|
||
#ifdef TUI
|
||
#include "tui/tui.h" /* For tui_active et.al. */
|
||
#endif
|
||
|
||
#include <fcntl.h>
|
||
|
||
/* Prototypes for local command functions */
|
||
|
||
static void complete_command (char *, int);
|
||
|
||
static void echo_command (char *, int);
|
||
|
||
static void pwd_command (char *, int);
|
||
|
||
static void show_version (char *, int);
|
||
|
||
static void help_command (char *, int);
|
||
|
||
static void show_command (char *, int);
|
||
|
||
static void info_command (char *, int);
|
||
|
||
static void show_debug (char *, int);
|
||
|
||
static void set_debug (char *, int);
|
||
|
||
static void show_user (char *, int);
|
||
|
||
static void make_command (char *, int);
|
||
|
||
static void shell_escape (char *, int);
|
||
|
||
static void edit_command (char *, int);
|
||
|
||
static void list_command (char *, int);
|
||
|
||
/* Prototypes for local utility functions */
|
||
|
||
static void ambiguous_line_spec (struct symtabs_and_lines *);
|
||
|
||
static void filter_sals (struct symtabs_and_lines *);
|
||
|
||
|
||
/* Limit the call depth of user-defined commands */
|
||
unsigned int max_user_call_depth;
|
||
|
||
/* Define all cmd_list_elements. */
|
||
|
||
/* Chain containing all defined commands. */
|
||
|
||
struct cmd_list_element *cmdlist;
|
||
|
||
/* Chain containing all defined info subcommands. */
|
||
|
||
struct cmd_list_element *infolist;
|
||
|
||
/* Chain containing all defined enable subcommands. */
|
||
|
||
struct cmd_list_element *enablelist;
|
||
|
||
/* Chain containing all defined disable subcommands. */
|
||
|
||
struct cmd_list_element *disablelist;
|
||
|
||
/* Chain containing all defined stop subcommands. */
|
||
|
||
struct cmd_list_element *stoplist;
|
||
|
||
/* Chain containing all defined delete subcommands. */
|
||
|
||
struct cmd_list_element *deletelist;
|
||
|
||
/* Chain containing all defined detach subcommands. */
|
||
|
||
struct cmd_list_element *detachlist;
|
||
|
||
/* Chain containing all defined kill subcommands. */
|
||
|
||
struct cmd_list_element *killlist;
|
||
|
||
/* Chain containing all defined set subcommands */
|
||
|
||
struct cmd_list_element *setlist;
|
||
|
||
/* Chain containing all defined unset subcommands */
|
||
|
||
struct cmd_list_element *unsetlist;
|
||
|
||
/* Chain containing all defined show subcommands. */
|
||
|
||
struct cmd_list_element *showlist;
|
||
|
||
/* Chain containing all defined \"set history\". */
|
||
|
||
struct cmd_list_element *sethistlist;
|
||
|
||
/* Chain containing all defined \"show history\". */
|
||
|
||
struct cmd_list_element *showhistlist;
|
||
|
||
/* Chain containing all defined \"unset history\". */
|
||
|
||
struct cmd_list_element *unsethistlist;
|
||
|
||
/* Chain containing all defined maintenance subcommands. */
|
||
|
||
struct cmd_list_element *maintenancelist;
|
||
|
||
/* Chain containing all defined "maintenance info" subcommands. */
|
||
|
||
struct cmd_list_element *maintenanceinfolist;
|
||
|
||
/* Chain containing all defined "maintenance print" subcommands. */
|
||
|
||
struct cmd_list_element *maintenanceprintlist;
|
||
|
||
struct cmd_list_element *setprintlist;
|
||
|
||
struct cmd_list_element *showprintlist;
|
||
|
||
struct cmd_list_element *setdebuglist;
|
||
|
||
struct cmd_list_element *showdebuglist;
|
||
|
||
struct cmd_list_element *setchecklist;
|
||
|
||
struct cmd_list_element *showchecklist;
|
||
|
||
/* Command tracing state. */
|
||
|
||
int source_verbose = 0;
|
||
int trace_commands = 0;
|
||
|
||
/* 'script-extension' option support. */
|
||
|
||
static const char script_ext_off[] = "off";
|
||
static const char script_ext_soft[] = "soft";
|
||
static const char script_ext_strict[] = "strict";
|
||
|
||
static const char *const script_ext_enums[] = {
|
||
script_ext_off,
|
||
script_ext_soft,
|
||
script_ext_strict,
|
||
NULL
|
||
};
|
||
|
||
static const char *script_ext_mode = script_ext_soft;
|
||
|
||
/* Utility used everywhere when at least one argument is needed and
|
||
none is supplied. */
|
||
|
||
void
|
||
error_no_arg (const char *why)
|
||
{
|
||
error (_("Argument required (%s)."), why);
|
||
}
|
||
|
||
/* The "info" command is defined as a prefix, with allow_unknown = 0.
|
||
Therefore, its own definition is called only for "info" with no
|
||
args. */
|
||
|
||
static void
|
||
info_command (char *arg, int from_tty)
|
||
{
|
||
printf_unfiltered (_("\"info\" must be followed by "
|
||
"the name of an info command.\n"));
|
||
help_list (infolist, "info ", all_commands, gdb_stdout);
|
||
}
|
||
|
||
/* The "show" command with no arguments shows all the settings. */
|
||
|
||
static void
|
||
show_command (char *arg, int from_tty)
|
||
{
|
||
cmd_show_list (showlist, from_tty, "");
|
||
}
|
||
|
||
/* Provide documentation on command or list given by COMMAND. FROM_TTY
|
||
is ignored. */
|
||
|
||
static void
|
||
help_command (char *command, int from_tty)
|
||
{
|
||
help_cmd (command, gdb_stdout);
|
||
}
|
||
|
||
/* Note: The "complete" command is used by Emacs to implement completion.
|
||
[Is that why this function writes output with *_unfiltered?] */
|
||
|
||
static void
|
||
complete_command (char *arg, int from_tty)
|
||
{
|
||
int argpoint;
|
||
char *point, *arg_prefix;
|
||
VEC (char_ptr) *completions;
|
||
|
||
dont_repeat ();
|
||
|
||
if (max_completions == 0)
|
||
{
|
||
/* Only print this for non-mi frontends. An MI frontend may not
|
||
be able to handle this. */
|
||
if (!ui_out_is_mi_like_p (current_uiout))
|
||
{
|
||
printf_unfiltered (_("max-completions is zero,"
|
||
" completion is disabled.\n"));
|
||
}
|
||
return;
|
||
}
|
||
|
||
if (arg == NULL)
|
||
arg = "";
|
||
argpoint = strlen (arg);
|
||
|
||
/* complete_line assumes that its first argument is somewhere
|
||
within, and except for filenames at the beginning of, the word to
|
||
be completed. The following crude imitation of readline's
|
||
word-breaking tries to accomodate this. */
|
||
point = arg + argpoint;
|
||
while (point > arg)
|
||
{
|
||
if (strchr (rl_completer_word_break_characters, point[-1]) != 0)
|
||
break;
|
||
point--;
|
||
}
|
||
|
||
arg_prefix = (char *) alloca (point - arg + 1);
|
||
memcpy (arg_prefix, arg, point - arg);
|
||
arg_prefix[point - arg] = 0;
|
||
|
||
completions = complete_line (point, arg, argpoint);
|
||
|
||
if (completions)
|
||
{
|
||
int ix, size = VEC_length (char_ptr, completions);
|
||
char *item, *prev = NULL;
|
||
|
||
qsort (VEC_address (char_ptr, completions), size,
|
||
sizeof (char *), compare_strings);
|
||
|
||
/* We do extra processing here since we only want to print each
|
||
unique item once. */
|
||
for (ix = 0; VEC_iterate (char_ptr, completions, ix, item); ++ix)
|
||
{
|
||
if (prev == NULL || strcmp (item, prev) != 0)
|
||
{
|
||
printf_unfiltered ("%s%s\n", arg_prefix, item);
|
||
xfree (prev);
|
||
prev = item;
|
||
}
|
||
else
|
||
xfree (item);
|
||
}
|
||
|
||
xfree (prev);
|
||
VEC_free (char_ptr, completions);
|
||
|
||
if (size == max_completions)
|
||
{
|
||
/* ARG_PREFIX and POINT are included in the output so that emacs
|
||
will include the message in the output. */
|
||
printf_unfiltered (_("%s%s %s\n"),
|
||
arg_prefix, point,
|
||
get_max_completions_reached_message ());
|
||
}
|
||
}
|
||
}
|
||
|
||
int
|
||
is_complete_command (struct cmd_list_element *c)
|
||
{
|
||
return cmd_cfunc_eq (c, complete_command);
|
||
}
|
||
|
||
static void
|
||
show_version (char *args, int from_tty)
|
||
{
|
||
print_gdb_version (gdb_stdout);
|
||
printf_filtered ("\n");
|
||
}
|
||
|
||
static void
|
||
show_configuration (char *args, int from_tty)
|
||
{
|
||
print_gdb_configuration (gdb_stdout);
|
||
}
|
||
|
||
/* Handle the quit command. */
|
||
|
||
void
|
||
quit_command (char *args, int from_tty)
|
||
{
|
||
if (!quit_confirm ())
|
||
error (_("Not confirmed."));
|
||
|
||
query_if_trace_running (from_tty);
|
||
|
||
quit_force (args, from_tty);
|
||
}
|
||
|
||
static void
|
||
pwd_command (char *args, int from_tty)
|
||
{
|
||
if (args)
|
||
error (_("The \"pwd\" command does not take an argument: %s"), args);
|
||
if (! getcwd (gdb_dirbuf, sizeof (gdb_dirbuf)))
|
||
error (_("Error finding name of working directory: %s"),
|
||
safe_strerror (errno));
|
||
|
||
if (strcmp (gdb_dirbuf, current_directory) != 0)
|
||
printf_unfiltered (_("Working directory %s\n (canonically %s).\n"),
|
||
current_directory, gdb_dirbuf);
|
||
else
|
||
printf_unfiltered (_("Working directory %s.\n"), current_directory);
|
||
}
|
||
|
||
void
|
||
cd_command (char *dir, int from_tty)
|
||
{
|
||
int len;
|
||
/* Found something other than leading repetitions of "/..". */
|
||
int found_real_path;
|
||
char *p;
|
||
struct cleanup *cleanup;
|
||
|
||
/* If the new directory is absolute, repeat is a no-op; if relative,
|
||
repeat might be useful but is more likely to be a mistake. */
|
||
dont_repeat ();
|
||
|
||
if (dir == 0)
|
||
dir = "~";
|
||
|
||
dir = tilde_expand (dir);
|
||
cleanup = make_cleanup (xfree, dir);
|
||
|
||
if (chdir (dir) < 0)
|
||
perror_with_name (dir);
|
||
|
||
#ifdef HAVE_DOS_BASED_FILE_SYSTEM
|
||
/* There's too much mess with DOSish names like "d:", "d:.",
|
||
"d:./foo" etc. Instead of having lots of special #ifdef'ed code,
|
||
simply get the canonicalized name of the current directory. */
|
||
dir = getcwd (gdb_dirbuf, sizeof (gdb_dirbuf));
|
||
#endif
|
||
|
||
len = strlen (dir);
|
||
if (IS_DIR_SEPARATOR (dir[len - 1]))
|
||
{
|
||
/* Remove the trailing slash unless this is a root directory
|
||
(including a drive letter on non-Unix systems). */
|
||
if (!(len == 1) /* "/" */
|
||
#ifdef HAVE_DOS_BASED_FILE_SYSTEM
|
||
&& !(len == 3 && dir[1] == ':') /* "d:/" */
|
||
#endif
|
||
)
|
||
len--;
|
||
}
|
||
|
||
dir = savestring (dir, len);
|
||
if (IS_ABSOLUTE_PATH (dir))
|
||
current_directory = dir;
|
||
else
|
||
{
|
||
if (IS_DIR_SEPARATOR (current_directory[strlen (current_directory) - 1]))
|
||
current_directory = concat (current_directory, dir, (char *)NULL);
|
||
else
|
||
current_directory = concat (current_directory, SLASH_STRING,
|
||
dir, (char *)NULL);
|
||
xfree (dir);
|
||
}
|
||
|
||
/* Now simplify any occurrences of `.' and `..' in the pathname. */
|
||
|
||
found_real_path = 0;
|
||
for (p = current_directory; *p;)
|
||
{
|
||
if (IS_DIR_SEPARATOR (p[0]) && p[1] == '.'
|
||
&& (p[2] == 0 || IS_DIR_SEPARATOR (p[2])))
|
||
memmove (p, p + 2, strlen (p + 2) + 1);
|
||
else if (IS_DIR_SEPARATOR (p[0]) && p[1] == '.' && p[2] == '.'
|
||
&& (p[3] == 0 || IS_DIR_SEPARATOR (p[3])))
|
||
{
|
||
if (found_real_path)
|
||
{
|
||
/* Search backwards for the directory just before the "/.."
|
||
and obliterate it and the "/..". */
|
||
char *q = p;
|
||
|
||
while (q != current_directory && !IS_DIR_SEPARATOR (q[-1]))
|
||
--q;
|
||
|
||
if (q == current_directory)
|
||
/* current_directory is
|
||
a relative pathname ("can't happen"--leave it alone). */
|
||
++p;
|
||
else
|
||
{
|
||
memmove (q - 1, p + 3, strlen (p + 3) + 1);
|
||
p = q - 1;
|
||
}
|
||
}
|
||
else
|
||
/* We are dealing with leading repetitions of "/..", for
|
||
example "/../..", which is the Mach super-root. */
|
||
p += 3;
|
||
}
|
||
else
|
||
{
|
||
found_real_path = 1;
|
||
++p;
|
||
}
|
||
}
|
||
|
||
forget_cached_source_info ();
|
||
|
||
if (from_tty)
|
||
pwd_command ((char *) 0, 1);
|
||
|
||
do_cleanups (cleanup);
|
||
}
|
||
|
||
/* Show the current value of the 'script-extension' option. */
|
||
|
||
static void
|
||
show_script_ext_mode (struct ui_file *file, int from_tty,
|
||
struct cmd_list_element *c, const char *value)
|
||
{
|
||
fprintf_filtered (file,
|
||
_("Script filename extension recognition is \"%s\".\n"),
|
||
value);
|
||
}
|
||
|
||
/* Try to open SCRIPT_FILE.
|
||
If successful, the full path name is stored in *FULL_PATHP,
|
||
the stream is stored in *STREAMP, and return 1.
|
||
The caller is responsible for freeing *FULL_PATHP.
|
||
If not successful, return 0; errno is set for the last file
|
||
we tried to open.
|
||
|
||
If SEARCH_PATH is non-zero, and the file isn't found in cwd,
|
||
search for it in the source search path. */
|
||
|
||
int
|
||
find_and_open_script (const char *script_file, int search_path,
|
||
FILE **streamp, char **full_pathp)
|
||
{
|
||
char *file;
|
||
int fd;
|
||
struct cleanup *old_cleanups;
|
||
int search_flags = OPF_TRY_CWD_FIRST | OPF_RETURN_REALPATH;
|
||
|
||
file = tilde_expand (script_file);
|
||
old_cleanups = make_cleanup (xfree, file);
|
||
|
||
if (search_path)
|
||
search_flags |= OPF_SEARCH_IN_PATH;
|
||
|
||
/* Search for and open 'file' on the search path used for source
|
||
files. Put the full location in *FULL_PATHP. */
|
||
fd = openp (source_path, search_flags,
|
||
file, O_RDONLY, full_pathp);
|
||
|
||
if (fd == -1)
|
||
{
|
||
int save_errno = errno;
|
||
do_cleanups (old_cleanups);
|
||
errno = save_errno;
|
||
return 0;
|
||
}
|
||
|
||
do_cleanups (old_cleanups);
|
||
|
||
*streamp = fdopen (fd, FOPEN_RT);
|
||
if (*streamp == NULL)
|
||
{
|
||
int save_errno = errno;
|
||
|
||
close (fd);
|
||
if (full_pathp)
|
||
xfree (*full_pathp);
|
||
errno = save_errno;
|
||
return 0;
|
||
}
|
||
|
||
return 1;
|
||
}
|
||
|
||
/* Load script FILE, which has already been opened as STREAM. */
|
||
|
||
static void
|
||
source_script_from_stream (FILE *stream, const char *file)
|
||
{
|
||
if (script_ext_mode != script_ext_off)
|
||
{
|
||
const struct extension_language_defn *extlang
|
||
= get_ext_lang_of_file (file);
|
||
|
||
if (extlang != NULL)
|
||
{
|
||
if (ext_lang_present_p (extlang))
|
||
{
|
||
script_sourcer_func *sourcer
|
||
= ext_lang_script_sourcer (extlang);
|
||
|
||
gdb_assert (sourcer != NULL);
|
||
sourcer (extlang, stream, file);
|
||
return;
|
||
}
|
||
else if (script_ext_mode == script_ext_soft)
|
||
{
|
||
/* Assume the file is a gdb script.
|
||
This is handled below. */
|
||
}
|
||
else
|
||
throw_ext_lang_unsupported (extlang);
|
||
}
|
||
}
|
||
|
||
script_from_file (stream, file);
|
||
}
|
||
|
||
/* Worker to perform the "source" command.
|
||
Load script FILE.
|
||
If SEARCH_PATH is non-zero, and the file isn't found in cwd,
|
||
search for it in the source search path. */
|
||
|
||
static void
|
||
source_script_with_search (const char *file, int from_tty, int search_path)
|
||
{
|
||
FILE *stream;
|
||
char *full_path;
|
||
struct cleanup *old_cleanups;
|
||
|
||
if (file == NULL || *file == 0)
|
||
error (_("source command requires file name of file to source."));
|
||
|
||
if (!find_and_open_script (file, search_path, &stream, &full_path))
|
||
{
|
||
/* The script wasn't found, or was otherwise inaccessible.
|
||
If the source command was invoked interactively, throw an
|
||
error. Otherwise (e.g. if it was invoked by a script),
|
||
just emit a warning, rather than cause an error. */
|
||
if (from_tty)
|
||
perror_with_name (file);
|
||
else
|
||
{
|
||
perror_warning_with_name (file);
|
||
return;
|
||
}
|
||
}
|
||
|
||
old_cleanups = make_cleanup (xfree, full_path);
|
||
make_cleanup_fclose (stream);
|
||
/* The python support reopens the file, so we need to pass full_path here
|
||
in case the file was found on the search path. It's useful to do this
|
||
anyway so that error messages show the actual file used. But only do
|
||
this if we (may have) used search_path, as printing the full path in
|
||
errors for the non-search case can be more noise than signal. */
|
||
source_script_from_stream (stream, search_path ? full_path : file);
|
||
do_cleanups (old_cleanups);
|
||
}
|
||
|
||
/* Wrapper around source_script_with_search to export it to main.c
|
||
for use in loading .gdbinit scripts. */
|
||
|
||
void
|
||
source_script (const char *file, int from_tty)
|
||
{
|
||
source_script_with_search (file, from_tty, 0);
|
||
}
|
||
|
||
/* Return the source_verbose global variable to its previous state
|
||
on exit from the source command, by whatever means. */
|
||
static void
|
||
source_verbose_cleanup (void *old_value)
|
||
{
|
||
source_verbose = *(int *)old_value;
|
||
xfree (old_value);
|
||
}
|
||
|
||
static void
|
||
source_command (char *args, int from_tty)
|
||
{
|
||
struct cleanup *old_cleanups;
|
||
char *file = args;
|
||
int *old_source_verbose = XNEW (int);
|
||
int search_path = 0;
|
||
|
||
*old_source_verbose = source_verbose;
|
||
old_cleanups = make_cleanup (source_verbose_cleanup,
|
||
old_source_verbose);
|
||
|
||
/* -v causes the source command to run in verbose mode.
|
||
-s causes the file to be searched in the source search path,
|
||
even if the file name contains a '/'.
|
||
We still have to be able to handle filenames with spaces in a
|
||
backward compatible way, so buildargv is not appropriate. */
|
||
|
||
if (args)
|
||
{
|
||
while (args[0] != '\0')
|
||
{
|
||
/* Make sure leading white space does not break the
|
||
comparisons. */
|
||
args = skip_spaces (args);
|
||
|
||
if (args[0] != '-')
|
||
break;
|
||
|
||
if (args[1] == 'v' && isspace (args[2]))
|
||
{
|
||
source_verbose = 1;
|
||
|
||
/* Skip passed -v. */
|
||
args = &args[3];
|
||
}
|
||
else if (args[1] == 's' && isspace (args[2]))
|
||
{
|
||
search_path = 1;
|
||
|
||
/* Skip passed -s. */
|
||
args = &args[3];
|
||
}
|
||
else
|
||
break;
|
||
}
|
||
|
||
file = skip_spaces (args);
|
||
}
|
||
|
||
source_script_with_search (file, from_tty, search_path);
|
||
|
||
do_cleanups (old_cleanups);
|
||
}
|
||
|
||
|
||
static void
|
||
echo_command (char *text, int from_tty)
|
||
{
|
||
const char *p = text;
|
||
int c;
|
||
|
||
if (text)
|
||
while ((c = *p++) != '\0')
|
||
{
|
||
if (c == '\\')
|
||
{
|
||
/* \ at end of argument is used after spaces
|
||
so they won't be lost. */
|
||
if (*p == 0)
|
||
return;
|
||
|
||
c = parse_escape (get_current_arch (), &p);
|
||
if (c >= 0)
|
||
printf_filtered ("%c", c);
|
||
}
|
||
else
|
||
printf_filtered ("%c", c);
|
||
}
|
||
|
||
/* Force this output to appear now. */
|
||
wrap_here ("");
|
||
gdb_flush (gdb_stdout);
|
||
}
|
||
|
||
static void
|
||
shell_escape (char *arg, int from_tty)
|
||
{
|
||
#if defined(CANT_FORK) || \
|
||
(!defined(HAVE_WORKING_VFORK) && !defined(HAVE_WORKING_FORK))
|
||
/* If ARG is NULL, they want an inferior shell, but `system' just
|
||
reports if the shell is available when passed a NULL arg. */
|
||
int rc = system (arg ? arg : "");
|
||
|
||
if (!arg)
|
||
arg = "inferior shell";
|
||
|
||
if (rc == -1)
|
||
{
|
||
fprintf_unfiltered (gdb_stderr, "Cannot execute %s: %s\n", arg,
|
||
safe_strerror (errno));
|
||
gdb_flush (gdb_stderr);
|
||
}
|
||
else if (rc)
|
||
{
|
||
fprintf_unfiltered (gdb_stderr, "%s exited with status %d\n", arg, rc);
|
||
gdb_flush (gdb_stderr);
|
||
}
|
||
#ifdef GLOBAL_CURDIR
|
||
/* Make sure to return to the directory GDB thinks it is, in case
|
||
the shell command we just ran changed it. */
|
||
chdir (current_directory);
|
||
#endif
|
||
#else /* Can fork. */
|
||
int status, pid;
|
||
|
||
if ((pid = vfork ()) == 0)
|
||
{
|
||
const char *p, *user_shell;
|
||
|
||
close_most_fds ();
|
||
|
||
if ((user_shell = (char *) getenv ("SHELL")) == NULL)
|
||
user_shell = "/bin/sh";
|
||
|
||
/* Get the name of the shell for arg0. */
|
||
p = lbasename (user_shell);
|
||
|
||
if (!arg)
|
||
execl (user_shell, p, (char *) 0);
|
||
else
|
||
execl (user_shell, p, "-c", arg, (char *) 0);
|
||
|
||
fprintf_unfiltered (gdb_stderr, "Cannot execute %s: %s\n", user_shell,
|
||
safe_strerror (errno));
|
||
gdb_flush (gdb_stderr);
|
||
_exit (0177);
|
||
}
|
||
|
||
if (pid != -1)
|
||
waitpid (pid, &status, 0);
|
||
else
|
||
error (_("Fork failed"));
|
||
#endif /* Can fork. */
|
||
}
|
||
|
||
static void
|
||
edit_command (char *arg, int from_tty)
|
||
{
|
||
struct symtabs_and_lines sals;
|
||
struct symtab_and_line sal;
|
||
struct symbol *sym;
|
||
char *editor;
|
||
char *p;
|
||
const char *fn;
|
||
|
||
/* Pull in the current default source line if necessary. */
|
||
if (arg == 0)
|
||
{
|
||
set_default_source_symtab_and_line ();
|
||
sal = get_current_source_symtab_and_line ();
|
||
}
|
||
|
||
/* Bare "edit" edits file with present line. */
|
||
|
||
if (arg == 0)
|
||
{
|
||
if (sal.symtab == 0)
|
||
error (_("No default source file yet."));
|
||
sal.line += get_lines_to_list () / 2;
|
||
}
|
||
else
|
||
{
|
||
struct cleanup *cleanup;
|
||
struct event_location *location;
|
||
char *arg1;
|
||
|
||
/* Now should only be one argument -- decode it in SAL. */
|
||
arg1 = arg;
|
||
location = string_to_event_location (&arg1, current_language);
|
||
cleanup = make_cleanup_delete_event_location (location);
|
||
sals = decode_line_1 (location, DECODE_LINE_LIST_MODE, 0, 0);
|
||
|
||
filter_sals (&sals);
|
||
if (! sals.nelts)
|
||
{
|
||
/* C++ */
|
||
do_cleanups (cleanup);
|
||
return;
|
||
}
|
||
if (sals.nelts > 1)
|
||
{
|
||
ambiguous_line_spec (&sals);
|
||
xfree (sals.sals);
|
||
do_cleanups (cleanup);
|
||
return;
|
||
}
|
||
|
||
sal = sals.sals[0];
|
||
xfree (sals.sals);
|
||
|
||
if (*arg1)
|
||
error (_("Junk at end of line specification."));
|
||
|
||
/* If line was specified by address, first print exactly which
|
||
line, and which file. In this case, sal.symtab == 0 means
|
||
address is outside of all known source files, not that user
|
||
failed to give a filename. */
|
||
if (*arg == '*')
|
||
{
|
||
struct gdbarch *gdbarch;
|
||
|
||
if (sal.symtab == 0)
|
||
error (_("No source file for address %s."),
|
||
paddress (get_current_arch (), sal.pc));
|
||
|
||
gdbarch = get_objfile_arch (SYMTAB_OBJFILE (sal.symtab));
|
||
sym = find_pc_function (sal.pc);
|
||
if (sym)
|
||
printf_filtered ("%s is in %s (%s:%d).\n",
|
||
paddress (gdbarch, sal.pc),
|
||
SYMBOL_PRINT_NAME (sym),
|
||
symtab_to_filename_for_display (sal.symtab),
|
||
sal.line);
|
||
else
|
||
printf_filtered ("%s is at %s:%d.\n",
|
||
paddress (gdbarch, sal.pc),
|
||
symtab_to_filename_for_display (sal.symtab),
|
||
sal.line);
|
||
}
|
||
|
||
/* If what was given does not imply a symtab, it must be an
|
||
undebuggable symbol which means no source code. */
|
||
|
||
if (sal.symtab == 0)
|
||
error (_("No line number known for %s."), arg);
|
||
do_cleanups (cleanup);
|
||
}
|
||
|
||
if ((editor = (char *) getenv ("EDITOR")) == NULL)
|
||
editor = "/bin/ex";
|
||
|
||
fn = symtab_to_fullname (sal.symtab);
|
||
|
||
/* Quote the file name, in case it has whitespace or other special
|
||
characters. */
|
||
p = xstrprintf ("%s +%d \"%s\"", editor, sal.line, fn);
|
||
shell_escape (p, from_tty);
|
||
xfree (p);
|
||
}
|
||
|
||
static void
|
||
list_command (char *arg, int from_tty)
|
||
{
|
||
struct symtabs_and_lines sals, sals_end;
|
||
struct symtab_and_line sal = { 0 };
|
||
struct symtab_and_line sal_end = { 0 };
|
||
struct symtab_and_line cursal = { 0 };
|
||
struct symbol *sym;
|
||
char *arg1;
|
||
int no_end = 1;
|
||
int dummy_end = 0;
|
||
int dummy_beg = 0;
|
||
int linenum_beg = 0;
|
||
char *p;
|
||
struct cleanup *cleanup;
|
||
|
||
cleanup = make_cleanup (null_cleanup, NULL);
|
||
|
||
/* Pull in the current default source line if necessary. */
|
||
if (arg == 0 || arg[0] == '+' || arg[0] == '-')
|
||
{
|
||
set_default_source_symtab_and_line ();
|
||
cursal = get_current_source_symtab_and_line ();
|
||
|
||
/* If this is the first "list" since we've set the current
|
||
source line, center the listing around that line. */
|
||
if (get_first_line_listed () == 0)
|
||
{
|
||
int first;
|
||
|
||
first = max (cursal.line - get_lines_to_list () / 2, 1);
|
||
|
||
/* A small special case --- if listing backwards, and we
|
||
should list only one line, list the preceding line,
|
||
instead of the exact line we've just shown after e.g.,
|
||
stopping for a breakpoint. */
|
||
if (arg != NULL && arg[0] == '-'
|
||
&& get_lines_to_list () == 1 && first > 1)
|
||
first -= 1;
|
||
|
||
print_source_lines (cursal.symtab, first,
|
||
first + get_lines_to_list (), 0);
|
||
return;
|
||
}
|
||
}
|
||
|
||
/* "l" or "l +" lists next ten lines. */
|
||
|
||
if (arg == 0 || strcmp (arg, "+") == 0)
|
||
{
|
||
print_source_lines (cursal.symtab, cursal.line,
|
||
cursal.line + get_lines_to_list (), 0);
|
||
return;
|
||
}
|
||
|
||
/* "l -" lists previous ten lines, the ones before the ten just
|
||
listed. */
|
||
if (strcmp (arg, "-") == 0)
|
||
{
|
||
print_source_lines (cursal.symtab,
|
||
max (get_first_line_listed ()
|
||
- get_lines_to_list (), 1),
|
||
get_first_line_listed (), 0);
|
||
return;
|
||
}
|
||
|
||
/* Now if there is only one argument, decode it in SAL
|
||
and set NO_END.
|
||
If there are two arguments, decode them in SAL and SAL_END
|
||
and clear NO_END; however, if one of the arguments is blank,
|
||
set DUMMY_BEG or DUMMY_END to record that fact. */
|
||
|
||
if (!have_full_symbols () && !have_partial_symbols ())
|
||
error (_("No symbol table is loaded. Use the \"file\" command."));
|
||
|
||
arg1 = arg;
|
||
if (*arg1 == ',')
|
||
dummy_beg = 1;
|
||
else
|
||
{
|
||
struct event_location *location;
|
||
|
||
location = string_to_event_location (&arg1, current_language);
|
||
make_cleanup_delete_event_location (location);
|
||
sals = decode_line_1 (location, DECODE_LINE_LIST_MODE, 0, 0);
|
||
|
||
filter_sals (&sals);
|
||
if (!sals.nelts)
|
||
{
|
||
/* C++ */
|
||
do_cleanups (cleanup);
|
||
return;
|
||
}
|
||
if (sals.nelts > 1)
|
||
{
|
||
ambiguous_line_spec (&sals);
|
||
xfree (sals.sals);
|
||
do_cleanups (cleanup);
|
||
return;
|
||
}
|
||
|
||
sal = sals.sals[0];
|
||
xfree (sals.sals);
|
||
}
|
||
|
||
/* Record whether the BEG arg is all digits. */
|
||
|
||
for (p = arg; p != arg1 && *p >= '0' && *p <= '9'; p++);
|
||
linenum_beg = (p == arg1);
|
||
|
||
while (*arg1 == ' ' || *arg1 == '\t')
|
||
arg1++;
|
||
if (*arg1 == ',')
|
||
{
|
||
no_end = 0;
|
||
arg1++;
|
||
while (*arg1 == ' ' || *arg1 == '\t')
|
||
arg1++;
|
||
if (*arg1 == 0)
|
||
dummy_end = 1;
|
||
else
|
||
{
|
||
struct event_location *location;
|
||
|
||
location = string_to_event_location (&arg1, current_language);
|
||
make_cleanup_delete_event_location (location);
|
||
if (dummy_beg)
|
||
sals_end = decode_line_1 (location,
|
||
DECODE_LINE_LIST_MODE, 0, 0);
|
||
else
|
||
sals_end = decode_line_1 (location, DECODE_LINE_LIST_MODE,
|
||
sal.symtab, sal.line);
|
||
|
||
filter_sals (&sals_end);
|
||
if (sals_end.nelts == 0)
|
||
{
|
||
do_cleanups (cleanup);
|
||
return;
|
||
}
|
||
if (sals_end.nelts > 1)
|
||
{
|
||
ambiguous_line_spec (&sals_end);
|
||
xfree (sals_end.sals);
|
||
do_cleanups (cleanup);
|
||
return;
|
||
}
|
||
sal_end = sals_end.sals[0];
|
||
xfree (sals_end.sals);
|
||
}
|
||
}
|
||
|
||
if (*arg1)
|
||
error (_("Junk at end of line specification."));
|
||
|
||
if (!no_end && !dummy_beg && !dummy_end
|
||
&& sal.symtab != sal_end.symtab)
|
||
error (_("Specified start and end are in different files."));
|
||
if (dummy_beg && dummy_end)
|
||
error (_("Two empty args do not say what lines to list."));
|
||
|
||
/* If line was specified by address,
|
||
first print exactly which line, and which file.
|
||
|
||
In this case, sal.symtab == 0 means address is outside of all
|
||
known source files, not that user failed to give a filename. */
|
||
if (*arg == '*')
|
||
{
|
||
struct gdbarch *gdbarch;
|
||
|
||
if (sal.symtab == 0)
|
||
error (_("No source file for address %s."),
|
||
paddress (get_current_arch (), sal.pc));
|
||
|
||
gdbarch = get_objfile_arch (SYMTAB_OBJFILE (sal.symtab));
|
||
sym = find_pc_function (sal.pc);
|
||
if (sym)
|
||
printf_filtered ("%s is in %s (%s:%d).\n",
|
||
paddress (gdbarch, sal.pc),
|
||
SYMBOL_PRINT_NAME (sym),
|
||
symtab_to_filename_for_display (sal.symtab), sal.line);
|
||
else
|
||
printf_filtered ("%s is at %s:%d.\n",
|
||
paddress (gdbarch, sal.pc),
|
||
symtab_to_filename_for_display (sal.symtab), sal.line);
|
||
}
|
||
|
||
/* If line was not specified by just a line number, and it does not
|
||
imply a symtab, it must be an undebuggable symbol which means no
|
||
source code. */
|
||
|
||
if (!linenum_beg && sal.symtab == 0)
|
||
error (_("No line number known for %s."), arg);
|
||
|
||
/* If this command is repeated with RET,
|
||
turn it into the no-arg variant. */
|
||
|
||
if (from_tty)
|
||
*arg = 0;
|
||
|
||
if (dummy_beg && sal_end.symtab == 0)
|
||
error (_("No default source file yet. Do \"help list\"."));
|
||
if (dummy_beg)
|
||
print_source_lines (sal_end.symtab,
|
||
max (sal_end.line - (get_lines_to_list () - 1), 1),
|
||
sal_end.line + 1, 0);
|
||
else if (sal.symtab == 0)
|
||
error (_("No default source file yet. Do \"help list\"."));
|
||
else if (no_end)
|
||
{
|
||
int first_line = sal.line - get_lines_to_list () / 2;
|
||
|
||
if (first_line < 1) first_line = 1;
|
||
|
||
print_source_lines (sal.symtab,
|
||
first_line,
|
||
first_line + get_lines_to_list (),
|
||
0);
|
||
}
|
||
else
|
||
print_source_lines (sal.symtab, sal.line,
|
||
(dummy_end
|
||
? sal.line + get_lines_to_list ()
|
||
: sal_end.line + 1),
|
||
0);
|
||
do_cleanups (cleanup);
|
||
}
|
||
|
||
/* Subroutine of disassemble_command to simplify it.
|
||
Perform the disassembly.
|
||
NAME is the name of the function if known, or NULL.
|
||
[LOW,HIGH) are the range of addresses to disassemble.
|
||
MIXED is non-zero to print source with the assembler. */
|
||
|
||
static void
|
||
print_disassembly (struct gdbarch *gdbarch, const char *name,
|
||
CORE_ADDR low, CORE_ADDR high, int flags)
|
||
{
|
||
#if defined(TUI)
|
||
if (!tui_is_window_visible (DISASSEM_WIN))
|
||
#endif
|
||
{
|
||
printf_filtered ("Dump of assembler code ");
|
||
if (name != NULL)
|
||
printf_filtered ("for function %s:\n", name);
|
||
else
|
||
printf_filtered ("from %s to %s:\n",
|
||
paddress (gdbarch, low), paddress (gdbarch, high));
|
||
|
||
/* Dump the specified range. */
|
||
gdb_disassembly (gdbarch, current_uiout, 0, flags, -1, low, high);
|
||
|
||
printf_filtered ("End of assembler dump.\n");
|
||
gdb_flush (gdb_stdout);
|
||
}
|
||
#if defined(TUI)
|
||
else
|
||
{
|
||
tui_show_assembly (gdbarch, low);
|
||
}
|
||
#endif
|
||
}
|
||
|
||
/* Subroutine of disassemble_command to simplify it.
|
||
Print a disassembly of the current function according to FLAGS. */
|
||
|
||
static void
|
||
disassemble_current_function (int flags)
|
||
{
|
||
struct frame_info *frame;
|
||
struct gdbarch *gdbarch;
|
||
CORE_ADDR low, high, pc;
|
||
const char *name;
|
||
|
||
frame = get_selected_frame (_("No frame selected."));
|
||
gdbarch = get_frame_arch (frame);
|
||
pc = get_frame_address_in_block (frame);
|
||
if (find_pc_partial_function (pc, &name, &low, &high) == 0)
|
||
error (_("No function contains program counter for selected frame."));
|
||
#if defined(TUI)
|
||
/* NOTE: cagney/2003-02-13 The `tui_active' was previously
|
||
`tui_version'. */
|
||
if (tui_active)
|
||
/* FIXME: cagney/2004-02-07: This should be an observer. */
|
||
low = tui_get_low_disassembly_address (gdbarch, low, pc);
|
||
#endif
|
||
low += gdbarch_deprecated_function_start_offset (gdbarch);
|
||
|
||
print_disassembly (gdbarch, name, low, high, flags);
|
||
}
|
||
|
||
/* Dump a specified section of assembly code.
|
||
|
||
Usage:
|
||
disassemble [/mrs]
|
||
- dump the assembly code for the function of the current pc
|
||
disassemble [/mrs] addr
|
||
- dump the assembly code for the function at ADDR
|
||
disassemble [/mrs] low,high
|
||
disassemble [/mrs] low,+length
|
||
- dump the assembly code in the range [LOW,HIGH), or [LOW,LOW+length)
|
||
|
||
A /m modifier will include source code with the assembly in a
|
||
"source centric" view. This view lists only the file of the first insn,
|
||
even if other source files are involved (e.g., inlined functions), and
|
||
the output is in source order, even with optimized code. This view is
|
||
considered deprecated as it hasn't been useful in practice.
|
||
|
||
A /r modifier will include raw instructions in hex with the assembly.
|
||
|
||
A /s modifier will include source code with the assembly, like /m, with
|
||
two important differences:
|
||
1) The output is still in pc address order.
|
||
2) File names and contents for all relevant source files are displayed. */
|
||
|
||
static void
|
||
disassemble_command (char *arg, int from_tty)
|
||
{
|
||
struct gdbarch *gdbarch = get_current_arch ();
|
||
CORE_ADDR low, high;
|
||
const char *name;
|
||
CORE_ADDR pc;
|
||
int flags;
|
||
const char *p;
|
||
|
||
p = arg;
|
||
name = NULL;
|
||
flags = 0;
|
||
|
||
if (p && *p == '/')
|
||
{
|
||
++p;
|
||
|
||
if (*p == '\0')
|
||
error (_("Missing modifier."));
|
||
|
||
while (*p && ! isspace (*p))
|
||
{
|
||
switch (*p++)
|
||
{
|
||
case 'm':
|
||
flags |= DISASSEMBLY_SOURCE_DEPRECATED;
|
||
break;
|
||
case 'r':
|
||
flags |= DISASSEMBLY_RAW_INSN;
|
||
break;
|
||
case 's':
|
||
flags |= DISASSEMBLY_SOURCE;
|
||
break;
|
||
default:
|
||
error (_("Invalid disassembly modifier."));
|
||
}
|
||
}
|
||
|
||
p = skip_spaces_const (p);
|
||
}
|
||
|
||
if ((flags & (DISASSEMBLY_SOURCE_DEPRECATED | DISASSEMBLY_SOURCE))
|
||
== (DISASSEMBLY_SOURCE_DEPRECATED | DISASSEMBLY_SOURCE))
|
||
error (_("Cannot specify both /m and /s."));
|
||
|
||
if (! p || ! *p)
|
||
{
|
||
flags |= DISASSEMBLY_OMIT_FNAME;
|
||
disassemble_current_function (flags);
|
||
return;
|
||
}
|
||
|
||
pc = value_as_address (parse_to_comma_and_eval (&p));
|
||
if (p[0] == ',')
|
||
++p;
|
||
if (p[0] == '\0')
|
||
{
|
||
/* One argument. */
|
||
if (find_pc_partial_function (pc, &name, &low, &high) == 0)
|
||
error (_("No function contains specified address."));
|
||
#if defined(TUI)
|
||
/* NOTE: cagney/2003-02-13 The `tui_active' was previously
|
||
`tui_version'. */
|
||
if (tui_active)
|
||
/* FIXME: cagney/2004-02-07: This should be an observer. */
|
||
low = tui_get_low_disassembly_address (gdbarch, low, pc);
|
||
#endif
|
||
low += gdbarch_deprecated_function_start_offset (gdbarch);
|
||
flags |= DISASSEMBLY_OMIT_FNAME;
|
||
}
|
||
else
|
||
{
|
||
/* Two arguments. */
|
||
int incl_flag = 0;
|
||
low = pc;
|
||
p = skip_spaces_const (p);
|
||
if (p[0] == '+')
|
||
{
|
||
++p;
|
||
incl_flag = 1;
|
||
}
|
||
high = parse_and_eval_address (p);
|
||
if (incl_flag)
|
||
high += low;
|
||
}
|
||
|
||
print_disassembly (gdbarch, name, low, high, flags);
|
||
}
|
||
|
||
static void
|
||
make_command (char *arg, int from_tty)
|
||
{
|
||
char *p;
|
||
|
||
if (arg == 0)
|
||
p = "make";
|
||
else
|
||
{
|
||
p = (char *) xmalloc (sizeof ("make ") + strlen (arg));
|
||
strcpy (p, "make ");
|
||
strcpy (p + sizeof ("make ") - 1, arg);
|
||
}
|
||
|
||
shell_escape (p, from_tty);
|
||
}
|
||
|
||
static void
|
||
show_user (char *args, int from_tty)
|
||
{
|
||
struct cmd_list_element *c;
|
||
extern struct cmd_list_element *cmdlist;
|
||
|
||
if (args)
|
||
{
|
||
const char *comname = args;
|
||
|
||
c = lookup_cmd (&comname, cmdlist, "", 0, 1);
|
||
if (!cli_user_command_p (c))
|
||
error (_("Not a user command."));
|
||
show_user_1 (c, "", args, gdb_stdout);
|
||
}
|
||
else
|
||
{
|
||
for (c = cmdlist; c; c = c->next)
|
||
{
|
||
if (cli_user_command_p (c) || c->prefixlist != NULL)
|
||
show_user_1 (c, "", c->name, gdb_stdout);
|
||
}
|
||
}
|
||
}
|
||
|
||
/* Search through names of commands and documentations for a certain
|
||
regular expression. */
|
||
|
||
static void
|
||
apropos_command (char *searchstr, int from_tty)
|
||
{
|
||
regex_t pattern;
|
||
int code;
|
||
|
||
if (searchstr == NULL)
|
||
error (_("REGEXP string is empty"));
|
||
|
||
code = regcomp (&pattern, searchstr, REG_ICASE);
|
||
if (code == 0)
|
||
{
|
||
struct cleanup *cleanups;
|
||
|
||
cleanups = make_regfree_cleanup (&pattern);
|
||
apropos_cmd (gdb_stdout, cmdlist, &pattern, "");
|
||
do_cleanups (cleanups);
|
||
}
|
||
else
|
||
{
|
||
char *err = get_regcomp_error (code, &pattern);
|
||
|
||
make_cleanup (xfree, err);
|
||
error (_("Error in regular expression: %s"), err);
|
||
}
|
||
}
|
||
|
||
/* Subroutine of alias_command to simplify it.
|
||
Return the first N elements of ARGV flattened back to a string
|
||
with a space separating each element.
|
||
ARGV may not be NULL.
|
||
This does not take care of quoting elements in case they contain spaces
|
||
on purpose. */
|
||
|
||
static dyn_string_t
|
||
argv_to_dyn_string (char **argv, int n)
|
||
{
|
||
int i;
|
||
dyn_string_t result = dyn_string_new (10);
|
||
|
||
gdb_assert (argv != NULL);
|
||
gdb_assert (n >= 0 && n <= countargv (argv));
|
||
|
||
for (i = 0; i < n; ++i)
|
||
{
|
||
if (i > 0)
|
||
dyn_string_append_char (result, ' ');
|
||
dyn_string_append_cstr (result, argv[i]);
|
||
}
|
||
|
||
return result;
|
||
}
|
||
|
||
/* Subroutine of alias_command to simplify it.
|
||
Return TRUE if COMMAND exists, unambiguously. Otherwise FALSE. */
|
||
|
||
static int
|
||
valid_command_p (const char *command)
|
||
{
|
||
struct cmd_list_element *c;
|
||
|
||
c = lookup_cmd_1 (& command, cmdlist, NULL, 1);
|
||
|
||
if (c == NULL || c == (struct cmd_list_element *) -1)
|
||
return FALSE;
|
||
|
||
/* This is the slightly tricky part.
|
||
lookup_cmd_1 will return a pointer to the last part of COMMAND
|
||
to match, leaving COMMAND pointing at the remainder. */
|
||
while (*command == ' ' || *command == '\t')
|
||
++command;
|
||
return *command == '\0';
|
||
}
|
||
|
||
/* Make an alias of an existing command. */
|
||
|
||
static void
|
||
alias_command (char *args, int from_tty)
|
||
{
|
||
int i, alias_argc, command_argc;
|
||
int abbrev_flag = 0;
|
||
char *args2, *equals, *alias, *command;
|
||
char **alias_argv, **command_argv;
|
||
dyn_string_t alias_dyn_string, command_dyn_string;
|
||
struct cleanup *cleanup;
|
||
static const char usage[] = N_("Usage: alias [-a] [--] ALIAS = COMMAND");
|
||
|
||
if (args == NULL || strchr (args, '=') == NULL)
|
||
error (_(usage));
|
||
|
||
args2 = xstrdup (args);
|
||
cleanup = make_cleanup (xfree, args2);
|
||
equals = strchr (args2, '=');
|
||
*equals = '\0';
|
||
alias_argv = gdb_buildargv (args2);
|
||
make_cleanup_freeargv (alias_argv);
|
||
command_argv = gdb_buildargv (equals + 1);
|
||
make_cleanup_freeargv (command_argv);
|
||
|
||
for (i = 0; alias_argv[i] != NULL; )
|
||
{
|
||
if (strcmp (alias_argv[i], "-a") == 0)
|
||
{
|
||
++alias_argv;
|
||
abbrev_flag = 1;
|
||
}
|
||
else if (strcmp (alias_argv[i], "--") == 0)
|
||
{
|
||
++alias_argv;
|
||
break;
|
||
}
|
||
else
|
||
break;
|
||
}
|
||
|
||
if (alias_argv[0] == NULL || command_argv[0] == NULL
|
||
|| *alias_argv[0] == '\0' || *command_argv[0] == '\0')
|
||
error (_(usage));
|
||
|
||
for (i = 0; alias_argv[i] != NULL; ++i)
|
||
{
|
||
if (! valid_user_defined_cmd_name_p (alias_argv[i]))
|
||
{
|
||
if (i == 0)
|
||
error (_("Invalid command name: %s"), alias_argv[i]);
|
||
else
|
||
error (_("Invalid command element name: %s"), alias_argv[i]);
|
||
}
|
||
}
|
||
|
||
alias_argc = countargv (alias_argv);
|
||
command_argc = countargv (command_argv);
|
||
|
||
/* COMMAND must exist.
|
||
Reconstruct the command to remove any extraneous spaces,
|
||
for better error messages. */
|
||
command_dyn_string = argv_to_dyn_string (command_argv, command_argc);
|
||
make_cleanup_dyn_string_delete (command_dyn_string);
|
||
command = dyn_string_buf (command_dyn_string);
|
||
if (! valid_command_p (command))
|
||
error (_("Invalid command to alias to: %s"), command);
|
||
|
||
/* ALIAS must not exist. */
|
||
alias_dyn_string = argv_to_dyn_string (alias_argv, alias_argc);
|
||
make_cleanup_dyn_string_delete (alias_dyn_string);
|
||
alias = dyn_string_buf (alias_dyn_string);
|
||
if (valid_command_p (alias))
|
||
error (_("Alias already exists: %s"), alias);
|
||
|
||
/* If ALIAS is one word, it is an alias for the entire COMMAND.
|
||
Example: alias spe = set print elements
|
||
|
||
Otherwise ALIAS and COMMAND must have the same number of words,
|
||
and every word except the last must match; and the last word of
|
||
ALIAS is made an alias of the last word of COMMAND.
|
||
Example: alias set print elms = set pr elem
|
||
Note that unambiguous abbreviations are allowed. */
|
||
|
||
if (alias_argc == 1)
|
||
{
|
||
/* add_cmd requires *we* allocate space for name, hence the xstrdup. */
|
||
add_com_alias (xstrdup (alias_argv[0]), command, class_alias,
|
||
abbrev_flag);
|
||
}
|
||
else
|
||
{
|
||
dyn_string_t alias_prefix_dyn_string, command_prefix_dyn_string;
|
||
const char *alias_prefix, *command_prefix;
|
||
struct cmd_list_element *c_alias, *c_command;
|
||
|
||
if (alias_argc != command_argc)
|
||
error (_("Mismatched command length between ALIAS and COMMAND."));
|
||
|
||
/* Create copies of ALIAS and COMMAND without the last word,
|
||
and use that to verify the leading elements match. */
|
||
alias_prefix_dyn_string =
|
||
argv_to_dyn_string (alias_argv, alias_argc - 1);
|
||
make_cleanup_dyn_string_delete (alias_prefix_dyn_string);
|
||
command_prefix_dyn_string =
|
||
argv_to_dyn_string (alias_argv, command_argc - 1);
|
||
make_cleanup_dyn_string_delete (command_prefix_dyn_string);
|
||
alias_prefix = dyn_string_buf (alias_prefix_dyn_string);
|
||
command_prefix = dyn_string_buf (command_prefix_dyn_string);
|
||
|
||
c_command = lookup_cmd_1 (& command_prefix, cmdlist, NULL, 1);
|
||
/* We've already tried to look up COMMAND. */
|
||
gdb_assert (c_command != NULL
|
||
&& c_command != (struct cmd_list_element *) -1);
|
||
gdb_assert (c_command->prefixlist != NULL);
|
||
c_alias = lookup_cmd_1 (& alias_prefix, cmdlist, NULL, 1);
|
||
if (c_alias != c_command)
|
||
error (_("ALIAS and COMMAND prefixes do not match."));
|
||
|
||
/* add_cmd requires *we* allocate space for name, hence the xstrdup. */
|
||
add_alias_cmd (xstrdup (alias_argv[alias_argc - 1]),
|
||
command_argv[command_argc - 1],
|
||
class_alias, abbrev_flag, c_command->prefixlist);
|
||
}
|
||
|
||
do_cleanups (cleanup);
|
||
}
|
||
|
||
/* Print a list of files and line numbers which a user may choose from
|
||
in order to list a function which was specified ambiguously (as
|
||
with `list classname::overloadedfuncname', for example). The
|
||
vector in SALS provides the filenames and line numbers. */
|
||
|
||
static void
|
||
ambiguous_line_spec (struct symtabs_and_lines *sals)
|
||
{
|
||
int i;
|
||
|
||
for (i = 0; i < sals->nelts; ++i)
|
||
printf_filtered (_("file: \"%s\", line number: %d\n"),
|
||
symtab_to_filename_for_display (sals->sals[i].symtab),
|
||
sals->sals[i].line);
|
||
}
|
||
|
||
/* Sort function for filter_sals. */
|
||
|
||
static int
|
||
compare_symtabs (const void *a, const void *b)
|
||
{
|
||
const struct symtab_and_line *sala = a;
|
||
const struct symtab_and_line *salb = b;
|
||
const char *dira = SYMTAB_DIRNAME (sala->symtab);
|
||
const char *dirb = SYMTAB_DIRNAME (salb->symtab);
|
||
int r;
|
||
|
||
if (dira == NULL)
|
||
{
|
||
if (dirb != NULL)
|
||
return -1;
|
||
}
|
||
else if (dirb == NULL)
|
||
{
|
||
if (dira != NULL)
|
||
return 1;
|
||
}
|
||
else
|
||
{
|
||
r = filename_cmp (dira, dirb);
|
||
if (r)
|
||
return r;
|
||
}
|
||
|
||
r = filename_cmp (sala->symtab->filename, salb->symtab->filename);
|
||
if (r)
|
||
return r;
|
||
|
||
if (sala->line < salb->line)
|
||
return -1;
|
||
return sala->line == salb->line ? 0 : 1;
|
||
}
|
||
|
||
/* Remove any SALs that do not match the current program space, or
|
||
which appear to be "file:line" duplicates. */
|
||
|
||
static void
|
||
filter_sals (struct symtabs_and_lines *sals)
|
||
{
|
||
int i, out, prev;
|
||
|
||
out = 0;
|
||
for (i = 0; i < sals->nelts; ++i)
|
||
{
|
||
if (sals->sals[i].pspace == current_program_space
|
||
&& sals->sals[i].symtab != NULL)
|
||
{
|
||
sals->sals[out] = sals->sals[i];
|
||
++out;
|
||
}
|
||
}
|
||
sals->nelts = out;
|
||
|
||
qsort (sals->sals, sals->nelts, sizeof (struct symtab_and_line),
|
||
compare_symtabs);
|
||
|
||
out = 1;
|
||
prev = 0;
|
||
for (i = 1; i < sals->nelts; ++i)
|
||
{
|
||
if (compare_symtabs (&sals->sals[prev], &sals->sals[i]))
|
||
{
|
||
/* Symtabs differ. */
|
||
sals->sals[out] = sals->sals[i];
|
||
prev = out;
|
||
++out;
|
||
}
|
||
}
|
||
|
||
if (sals->nelts == 0)
|
||
{
|
||
xfree (sals->sals);
|
||
sals->sals = NULL;
|
||
}
|
||
else
|
||
sals->nelts = out;
|
||
}
|
||
|
||
static void
|
||
set_debug (char *arg, int from_tty)
|
||
{
|
||
printf_unfiltered (_("\"set debug\" must be followed by "
|
||
"the name of a debug subcommand.\n"));
|
||
help_list (setdebuglist, "set debug ", all_commands, gdb_stdout);
|
||
}
|
||
|
||
static void
|
||
show_debug (char *args, int from_tty)
|
||
{
|
||
cmd_show_list (showdebuglist, from_tty, "");
|
||
}
|
||
|
||
void
|
||
init_cmd_lists (void)
|
||
{
|
||
max_user_call_depth = 1024;
|
||
}
|
||
|
||
static void
|
||
show_info_verbose (struct ui_file *file, int from_tty,
|
||
struct cmd_list_element *c,
|
||
const char *value)
|
||
{
|
||
if (info_verbose)
|
||
fprintf_filtered (file,
|
||
_("Verbose printing of informational messages is %s.\n"),
|
||
value);
|
||
else
|
||
fprintf_filtered (file, _("Verbosity is %s.\n"), value);
|
||
}
|
||
|
||
static void
|
||
show_history_expansion_p (struct ui_file *file, int from_tty,
|
||
struct cmd_list_element *c, const char *value)
|
||
{
|
||
fprintf_filtered (file, _("History expansion on command input is %s.\n"),
|
||
value);
|
||
}
|
||
|
||
static void
|
||
show_remote_debug (struct ui_file *file, int from_tty,
|
||
struct cmd_list_element *c, const char *value)
|
||
{
|
||
fprintf_filtered (file, _("Debugging of remote protocol is %s.\n"),
|
||
value);
|
||
}
|
||
|
||
static void
|
||
show_remote_timeout (struct ui_file *file, int from_tty,
|
||
struct cmd_list_element *c, const char *value)
|
||
{
|
||
fprintf_filtered (file,
|
||
_("Timeout limit to wait for target to respond is %s.\n"),
|
||
value);
|
||
}
|
||
|
||
static void
|
||
show_max_user_call_depth (struct ui_file *file, int from_tty,
|
||
struct cmd_list_element *c, const char *value)
|
||
{
|
||
fprintf_filtered (file,
|
||
_("The max call depth for user-defined commands is %s.\n"),
|
||
value);
|
||
}
|
||
|
||
|
||
|
||
initialize_file_ftype _initialize_cli_cmds;
|
||
|
||
void
|
||
_initialize_cli_cmds (void)
|
||
{
|
||
struct cmd_list_element *c;
|
||
|
||
/* Define the classes of commands.
|
||
They will appear in the help list in alphabetical order. */
|
||
|
||
add_cmd ("internals", class_maintenance, NULL, _("\
|
||
Maintenance commands.\n\
|
||
Some gdb commands are provided just for use by gdb maintainers.\n\
|
||
These commands are subject to frequent change, and may not be as\n\
|
||
well documented as user commands."),
|
||
&cmdlist);
|
||
add_cmd ("obscure", class_obscure, NULL, _("Obscure features."), &cmdlist);
|
||
add_cmd ("aliases", class_alias, NULL,
|
||
_("Aliases of other commands."), &cmdlist);
|
||
add_cmd ("user-defined", class_user, NULL, _("\
|
||
User-defined commands.\n\
|
||
The commands in this class are those defined by the user.\n\
|
||
Use the \"define\" command to define a command."), &cmdlist);
|
||
add_cmd ("support", class_support, NULL, _("Support facilities."), &cmdlist);
|
||
if (!dbx_commands)
|
||
add_cmd ("status", class_info, NULL, _("Status inquiries."), &cmdlist);
|
||
add_cmd ("files", class_files, NULL, _("Specifying and examining files."),
|
||
&cmdlist);
|
||
add_cmd ("breakpoints", class_breakpoint, NULL,
|
||
_("Making program stop at certain points."), &cmdlist);
|
||
add_cmd ("data", class_vars, NULL, _("Examining data."), &cmdlist);
|
||
add_cmd ("stack", class_stack, NULL, _("\
|
||
Examining the stack.\n\
|
||
The stack is made up of stack frames. Gdb assigns numbers to stack frames\n\
|
||
counting from zero for the innermost (currently executing) frame.\n\n\
|
||
At any time gdb identifies one frame as the \"selected\" frame.\n\
|
||
Variable lookups are done with respect to the selected frame.\n\
|
||
When the program being debugged stops, gdb selects the innermost frame.\n\
|
||
The commands below can be used to select other frames by number or address."),
|
||
&cmdlist);
|
||
add_cmd ("running", class_run, NULL, _("Running the program."), &cmdlist);
|
||
|
||
/* Define general commands. */
|
||
|
||
add_com ("pwd", class_files, pwd_command, _("\
|
||
Print working directory. This is used for your program as well."));
|
||
|
||
c = add_cmd ("cd", class_files, cd_command, _("\
|
||
Set working directory to DIR for debugger and program being debugged.\n\
|
||
The change does not take effect for the program being debugged\n\
|
||
until the next time it is started."), &cmdlist);
|
||
set_cmd_completer (c, filename_completer);
|
||
|
||
add_com ("echo", class_support, echo_command, _("\
|
||
Print a constant string. Give string as argument.\n\
|
||
C escape sequences may be used in the argument.\n\
|
||
No newline is added at the end of the argument;\n\
|
||
use \"\\n\" if you want a newline to be printed.\n\
|
||
Since leading and trailing whitespace are ignored in command arguments,\n\
|
||
if you want to print some you must use \"\\\" before leading whitespace\n\
|
||
to be printed or after trailing whitespace."));
|
||
|
||
add_setshow_enum_cmd ("script-extension", class_support,
|
||
script_ext_enums, &script_ext_mode, _("\
|
||
Set mode for script filename extension recognition."), _("\
|
||
Show mode for script filename extension recognition."), _("\
|
||
off == no filename extension recognition (all sourced files are GDB scripts)\n\
|
||
soft == evaluate script according to filename extension, fallback to GDB script"
|
||
"\n\
|
||
strict == evaluate script according to filename extension, error if not supported"
|
||
),
|
||
NULL,
|
||
show_script_ext_mode,
|
||
&setlist, &showlist);
|
||
|
||
add_com ("quit", class_support, quit_command, _("\
|
||
Exit gdb.\n\
|
||
Usage: quit [EXPR]\n\
|
||
The optional expression EXPR, if present, is evaluated and the result\n\
|
||
used as GDB's exit code. The default is zero."));
|
||
c = add_com ("help", class_support, help_command,
|
||
_("Print list of commands."));
|
||
set_cmd_completer (c, command_completer);
|
||
add_com_alias ("q", "quit", class_support, 1);
|
||
add_com_alias ("h", "help", class_support, 1);
|
||
|
||
add_setshow_boolean_cmd ("verbose", class_support, &info_verbose, _("\
|
||
Set verbosity."), _("\
|
||
Show verbosity."), NULL,
|
||
set_verbose,
|
||
show_info_verbose,
|
||
&setlist, &showlist);
|
||
|
||
add_prefix_cmd ("history", class_support, set_history,
|
||
_("Generic command for setting command history parameters."),
|
||
&sethistlist, "set history ", 0, &setlist);
|
||
add_prefix_cmd ("history", class_support, show_history,
|
||
_("Generic command for showing command history parameters."),
|
||
&showhistlist, "show history ", 0, &showlist);
|
||
|
||
add_setshow_boolean_cmd ("expansion", no_class, &history_expansion_p, _("\
|
||
Set history expansion on command input."), _("\
|
||
Show history expansion on command input."), _("\
|
||
Without an argument, history expansion is enabled."),
|
||
NULL,
|
||
show_history_expansion_p,
|
||
&sethistlist, &showhistlist);
|
||
|
||
add_prefix_cmd ("info", class_info, info_command, _("\
|
||
Generic command for showing things about the program being debugged."),
|
||
&infolist, "info ", 0, &cmdlist);
|
||
add_com_alias ("i", "info", class_info, 1);
|
||
add_com_alias ("inf", "info", class_info, 1);
|
||
|
||
add_com ("complete", class_obscure, complete_command,
|
||
_("List the completions for the rest of the line as a command."));
|
||
|
||
add_prefix_cmd ("show", class_info, show_command, _("\
|
||
Generic command for showing things about the debugger."),
|
||
&showlist, "show ", 0, &cmdlist);
|
||
/* Another way to get at the same thing. */
|
||
add_info ("set", show_command, _("Show all GDB settings."));
|
||
|
||
add_cmd ("commands", no_set_class, show_commands, _("\
|
||
Show the history of commands you typed.\n\
|
||
You can supply a command number to start with, or a `+' to start after\n\
|
||
the previous command number shown."),
|
||
&showlist);
|
||
|
||
add_cmd ("version", no_set_class, show_version,
|
||
_("Show what version of GDB this is."), &showlist);
|
||
|
||
add_cmd ("configuration", no_set_class, show_configuration,
|
||
_("Show how GDB was configured at build time."), &showlist);
|
||
|
||
add_setshow_zinteger_cmd ("remote", no_class, &remote_debug, _("\
|
||
Set debugging of remote protocol."), _("\
|
||
Show debugging of remote protocol."), _("\
|
||
When enabled, each packet sent or received with the remote target\n\
|
||
is displayed."),
|
||
NULL,
|
||
show_remote_debug,
|
||
&setdebuglist, &showdebuglist);
|
||
|
||
add_setshow_zuinteger_unlimited_cmd ("remotetimeout", no_class,
|
||
&remote_timeout, _("\
|
||
Set timeout limit to wait for target to respond."), _("\
|
||
Show timeout limit to wait for target to respond."), _("\
|
||
This value is used to set the time limit for gdb to wait for a response\n\
|
||
from the target."),
|
||
NULL,
|
||
show_remote_timeout,
|
||
&setlist, &showlist);
|
||
|
||
add_prefix_cmd ("debug", no_class, set_debug,
|
||
_("Generic command for setting gdb debugging flags"),
|
||
&setdebuglist, "set debug ", 0, &setlist);
|
||
|
||
add_prefix_cmd ("debug", no_class, show_debug,
|
||
_("Generic command for showing gdb debugging flags"),
|
||
&showdebuglist, "show debug ", 0, &showlist);
|
||
|
||
c = add_com ("shell", class_support, shell_escape, _("\
|
||
Execute the rest of the line as a shell command.\n\
|
||
With no arguments, run an inferior shell."));
|
||
set_cmd_completer (c, filename_completer);
|
||
|
||
c = add_com ("edit", class_files, edit_command, _("\
|
||
Edit specified file or function.\n\
|
||
With no argument, edits file containing most recent line listed.\n\
|
||
Editing targets can be specified in these ways:\n\
|
||
FILE:LINENUM, to edit at that line in that file,\n\
|
||
FUNCTION, to edit at the beginning of that function,\n\
|
||
FILE:FUNCTION, to distinguish among like-named static functions.\n\
|
||
*ADDRESS, to edit at the line containing that address.\n\
|
||
Uses EDITOR environment variable contents as editor (or ex as default)."));
|
||
|
||
c->completer = location_completer;
|
||
|
||
add_com ("list", class_files, list_command, _("\
|
||
List specified function or line.\n\
|
||
With no argument, lists ten more lines after or around previous listing.\n\
|
||
\"list -\" lists the ten lines before a previous ten-line listing.\n\
|
||
One argument specifies a line, and ten lines are listed around that line.\n\
|
||
Two arguments with comma between specify starting and ending lines to list.\n\
|
||
Lines can be specified in these ways:\n\
|
||
LINENUM, to list around that line in current file,\n\
|
||
FILE:LINENUM, to list around that line in that file,\n\
|
||
FUNCTION, to list around beginning of that function,\n\
|
||
FILE:FUNCTION, to distinguish among like-named static functions.\n\
|
||
*ADDRESS, to list around the line containing that address.\n\
|
||
With two args if one is empty it stands for ten lines away from \
|
||
the other arg."));
|
||
|
||
add_com_alias ("l", "list", class_files, 1);
|
||
|
||
if (dbx_commands)
|
||
add_com_alias ("file", "list", class_files, 1);
|
||
|
||
c = add_com ("disassemble", class_vars, disassemble_command, _("\
|
||
Disassemble a specified section of memory.\n\
|
||
Default is the function surrounding the pc of the selected frame.\n\
|
||
\n\
|
||
With a /m modifier, source lines are included (if available).\n\
|
||
This view is \"source centric\": the output is in source line order,\n\
|
||
regardless of any optimization that is present. Only the main source file\n\
|
||
is displayed, not those of, e.g., any inlined functions.\n\
|
||
This modifier hasn't proved useful in practice and is deprecated\n\
|
||
in favor of /s.\n\
|
||
\n\
|
||
With a /s modifier, source lines are included (if available).\n\
|
||
This differs from /m in two important respects:\n\
|
||
- the output is still in pc address order, and\n\
|
||
- file names and contents for all relevant source files are displayed.\n\
|
||
\n\
|
||
With a /r modifier, raw instructions in hex are included.\n\
|
||
\n\
|
||
With a single argument, the function surrounding that address is dumped.\n\
|
||
Two arguments (separated by a comma) are taken as a range of memory to dump,\n\
|
||
in the form of \"start,end\", or \"start,+length\".\n\
|
||
\n\
|
||
Note that the address is interpreted as an expression, not as a location\n\
|
||
like in the \"break\" command.\n\
|
||
So, for example, if you want to disassemble function bar in file foo.c\n\
|
||
you must type \"disassemble 'foo.c'::bar\" and not \"disassemble foo.c:bar\"."));
|
||
set_cmd_completer (c, location_completer);
|
||
|
||
add_com_alias ("!", "shell", class_support, 0);
|
||
|
||
c = add_com ("make", class_support, make_command, _("\
|
||
Run the ``make'' program using the rest of the line as arguments."));
|
||
set_cmd_completer (c, filename_completer);
|
||
add_cmd ("user", no_class, show_user, _("\
|
||
Show definitions of non-python/scheme user defined commands.\n\
|
||
Argument is the name of the user defined command.\n\
|
||
With no argument, show definitions of all user defined commands."), &showlist);
|
||
add_com ("apropos", class_support, apropos_command,
|
||
_("Search for commands matching a REGEXP"));
|
||
|
||
add_setshow_uinteger_cmd ("max-user-call-depth", no_class,
|
||
&max_user_call_depth, _("\
|
||
Set the max call depth for non-python/scheme user-defined commands."), _("\
|
||
Show the max call depth for non-python/scheme user-defined commands."), NULL,
|
||
NULL,
|
||
show_max_user_call_depth,
|
||
&setlist, &showlist);
|
||
|
||
add_setshow_boolean_cmd ("trace-commands", no_class, &trace_commands, _("\
|
||
Set tracing of GDB CLI commands."), _("\
|
||
Show state of GDB CLI command tracing."), _("\
|
||
When 'on', each command is displayed as it is executed."),
|
||
NULL,
|
||
NULL,
|
||
&setlist, &showlist);
|
||
|
||
c = add_com ("alias", class_support, alias_command, _("\
|
||
Define a new command that is an alias of an existing command.\n\
|
||
Usage: alias [-a] [--] ALIAS = COMMAND\n\
|
||
ALIAS is the name of the alias command to create.\n\
|
||
COMMAND is the command being aliased to.\n\
|
||
If \"-a\" is specified, the command is an abbreviation,\n\
|
||
and will not appear in help command list output.\n\
|
||
\n\
|
||
Examples:\n\
|
||
Make \"spe\" an alias of \"set print elements\":\n\
|
||
alias spe = set print elements\n\
|
||
Make \"elms\" an alias of \"elements\" in the \"set print\" command:\n\
|
||
alias -a set print elms = set print elements"));
|
||
}
|
||
|
||
void
|
||
init_cli_cmds (void)
|
||
{
|
||
struct cmd_list_element *c;
|
||
char *source_help_text;
|
||
|
||
source_help_text = xstrprintf (_("\
|
||
Read commands from a file named FILE.\n\
|
||
\n\
|
||
Usage: source [-s] [-v] FILE\n\
|
||
-s: search for the script in the source search path,\n\
|
||
even if FILE contains directories.\n\
|
||
-v: each command in FILE is echoed as it is executed.\n\
|
||
\n\
|
||
Note that the file \"%s\" is read automatically in this way\n\
|
||
when GDB is started."), gdbinit);
|
||
c = add_cmd ("source", class_support, source_command,
|
||
source_help_text, &cmdlist);
|
||
set_cmd_completer (c, filename_completer);
|
||
}
|