19ba03f495
See previous patch's description. gdb/ChangeLog: * macrocmd.c (print_macro_callback): Add cast(s). * macrotab.c (macro_bcache_str): Likewise. (new_macro_definition): Likewise. * main.c (captured_main): Likewise. * maint.c (print_bfd_section_info): Likewise. * mdebugread.c (mdebug_build_psymtabs): Likewise. (basic_type): Likewise. * memattr.c (mem_region_cmp): Likewise. * memory-map.c (memory_map_start_memory): Likewise. (memory_map_end_memory): Likewise. (memory_map_start_property): Likewise. (memory_map_end_property): Likewise. (clear_result): Likewise. * memrange.c (compare_mem_ranges): Likewise. * mep-tdep.c (mep_analyze_frame_prologue): Likewise. * mi/mi-cmd-var.c (mi_cmd_var_update_iter): Likewise. * mi/mi-console.c (mi_console_file_delete): Likewise. (mi_console_file_fputs): Likewise. (mi_console_raw_packet): Likewise. (mi_console_file_flush): Likewise. (mi_console_set_raw): Likewise. * mi/mi-interp.c (mi_interpreter_resume): Likewise. (mi_new_thread): Likewise. (mi_thread_exit): Likewise. (mi_record_changed): Likewise. (mi_inferior_added): Likewise. (mi_inferior_appeared): Likewise. (mi_inferior_exit): Likewise. (mi_inferior_removed): Likewise. (mi_interp_data): Likewise. (mi_on_normal_stop): Likewise. (mi_traceframe_changed): Likewise. (mi_tsv_created): Likewise. (mi_tsv_deleted): Likewise. (mi_tsv_modified): Likewise. (mi_breakpoint_created): Likewise. (mi_breakpoint_deleted): Likewise. (mi_breakpoint_modified): Likewise. (mi_output_running_pid): Likewise. (mi_inferior_count): Likewise. (mi_solib_loaded): Likewise. (mi_solib_unloaded): Likewise. (mi_command_param_changed): Likewise. (mi_memory_changed): Likewise. (report_initial_inferior): Likewise. (mi_ui_out): Likewise. (mi_set_logging): Likewise. * mi/mi-main.c (collect_cores): Likewise. (print_one_inferior): Likewise. (free_vector_of_ints): Likewise. (free_splay_tree): Likewise. (mi_execute_command): Likewise. * mi/mi-out.c (mi_table_body): Likewise. (mi_table_end): Likewise. (mi_table_header): Likewise. (mi_begin): Likewise. (mi_end): Likewise. (mi_field_int): Likewise. (mi_field_string): Likewise. (mi_field_fmt): Likewise. (mi_flush): Likewise. (mi_redirect): Likewise. (field_separator): Likewise. (mi_open): Likewise. (mi_close): Likewise. (mi_out_buffered): Likewise. (mi_out_rewind): Likewise. (mi_out_put): Likewise. (mi_version): Likewise. (mi_out_data_dtor): Likewise. * mi/mi-parse.c (mi_parse_cleanup): Likewise. * microblaze-tdep.c (microblaze_frame_cache): Likewise. * minidebug.c (lzma_open): Likewise. (lzma_pread): Likewise. (lzma_close): Likewise. (lzma_stat): Likewise. * mips-linux-tdep.c (mips_linux_init_abi): Likewise. * mips-sde-tdep.c (mips_sde_frame_cache): Likewise. (mips_sde_elf_osabi_sniff_abi_tag_sections): Likewise. * mips-tdep.c (mips_insn16_frame_cache): Likewise. (mips_micro_frame_cache): Likewise. (mips_insn32_frame_cache): Likewise. (mips_stub_frame_cache): Likewise. (gdb_print_insn_mips): Likewise. (value_of_mips_user_reg): Likewise. (mips_gdbarch_init): Likewise. * mips64obsd-tdep.c (mips64obsd_supply_gregset): Likewise. * mipsnbsd-tdep.c (mipsnbsd_supply_fpregset): Likewise. (mipsnbsd_supply_gregset): Likewise. * mn10300-linux-tdep.c (am33_supply_fpregset_method): Likewise. (am33_collect_gregset_method): Likewise. (am33_collect_fpregset_method): Likewise. * mn10300-tdep.c (mn10300_analyze_frame_prologue): Likewise. * moxie-tdep.c (moxie_frame_cache): Likewise. * msp430-tdep.c (msp430_get_opcode_byte): Likewise. (msp430_analyze_frame_prologue): Likewise. * mt-tdep.c (mt_frame_unwind_cache): Likewise. * nios2-linux-tdep.c (nios2_supply_gregset): Likewise. (nios2_collect_gregset): Likewise. * nios2-tdep.c (nios2_frame_unwind_cache): Likewise. (nios2_stub_frame_cache): Likewise. * objc-lang.c (find_methods): Likewise. * objfiles.c (objfiles_pspace_data_cleanup): Likewise. (get_objfile_pspace_data): Likewise. (get_objfile_bfd_data): Likewise. (objfile_bfd_data_free): Likewise. (add_to_objfile_sections): Likewise. (do_free_objfile_cleanup): Likewise. (resume_section_map_updates_cleanup): Likewise. * opencl-lang.c (builtin_opencl_type): Likewise. * osabi.c (generic_elf_osabi_sniff_abi_tag_sections): Likewise. * osdata.c (osdata_start_osdata): Likewise. (osdata_start_item): Likewise. (osdata_start_column): Likewise. (osdata_end_column): Likewise. (clear_parsing_data): Likewise. (osdata_free_cleanup): Likewise. * parse.c (type_stack_cleanup): Likewise. (exp_uses_objfile_iter): Likewise. * ppc-linux-tdep.c (ppc_linux_supply_gregset): Likewise. (ppc_linux_collect_gregset): Likewise. (ppu2spu_prev_arch): Likewise. (ppu2spu_this_id): Likewise. (ppu2spu_prev_register): Likewise. (ppu2spu_unwind_register): Likewise. (ppu2spu_sniffer): Likewise. (ppu2spu_dealloc_cache): Likewise. (ppc_linux_init_abi): Likewise. * ppcfbsd-tdep.c (ppcfbsd_sigtramp_frame_cache): Likewise. * ppcobsd-tdep.c (ppcobsd_sigtramp_frame_cache): Likewise. * progspace.c (restore_program_space): Likewise. * psymtab.c (find_pc_sect_psymtab): Likewise. (compare_psymbols): Likewise. (psymbol_bcache_full): Likewise. (allocate_psymtab): Likewise. (discard_psymtabs_upto): Likewise. * python/py-block.c (set_block): Likewise. (del_objfile_blocks): Likewise. * python/py-breakpoint.c (build_bp_list): Likewise. * python/py-inferior.c (inferior_to_inferior_object): Likewise. (build_inferior_list): Likewise. (py_free_inferior): Likewise. * python/py-objfile.c (py_free_objfile): Likewise. (objfile_to_objfile_object): Likewise. * python/py-prettyprint.c (py_restore_tstate): Likewise. * python/py-progspace.c (py_free_pspace): Likewise. (pspace_to_pspace_object): Likewise. * python/py-symbol.c (set_symbol): Likewise. (del_objfile_symbols): Likewise. * python/py-symtab.c (set_sal): Likewise. (set_symtab): Likewise. (del_objfile_symtab): Likewise. (del_objfile_sal): Likewise. * python/py-type.c (save_objfile_types): Likewise. (set_type): Likewise. * python/py-unwind.c (pyuw_prev_register): Likewise. (pyuw_on_new_gdbarch): Likewise. * python/py-utils.c (py_decref): Likewise. (py_xdecref): Likewise. (gdb_py_generic_dict): Likewise. * python/py-xmethods.c (gdbpy_free_xmethod_worker_data): Likewise. (gdbpy_clone_xmethod_worker_data): Likewise. (gdbpy_get_xmethod_arg_types): Likewise. (gdbpy_get_xmethod_result_type): Likewise. (gdbpy_invoke_xmethod): Likewise. * python/python.c (gdbpy_apply_type_printers): Likewise. (gdbpy_free_type_printers): Likewise. * record-btrace.c (record_btrace_disable_callback): Likewise. (bfcache_hash): Likewise. (bfcache_eq): Likewise. (btrace_get_frame_function): Likewise. (record_btrace_frame_unwind_stop_reason): Likewise. (record_btrace_frame_this_id): Likewise. (record_btrace_frame_prev_register): Likewise. (record_btrace_frame_dealloc_cache): Likewise. * record-full.c (record_full_message_wrapper): Likewise. (record_full_save_cleanups): Likewise. * regcache.c (regcache_descr): Likewise. (do_regcache_xfree): Likewise. (do_regcache_invalidate): Likewise. (do_cooked_read): Likewise. (regcache_transfer_regset): Likewise. * reggroups.c (reggroup_add): Likewise. (reggroup_next): Likewise. (reggroup_prev): Likewise. * remote-fileio.c (do_remote_fileio_request): Likewise. * remote-notif.c (remote_async_get_pending_events_handler): Likewise. (do_notif_event_xfree): Likewise. * remote.c (get_remote_arch_state): Likewise. (remote_pspace_data_cleanup): Likewise. (get_remote_exec_file): Likewise. (set_pspace_remote_exec_file): Likewise. (compare_pnums): Likewise. (clear_threads_listing_context): Likewise. (remote_newthread_step): Likewise. (start_thread): Likewise. (end_thread): Likewise. (remove_child_of_pending_fork): Likewise. (remove_stop_reply_for_inferior): Likewise. (remove_stop_reply_of_remote_state): Likewise. (remote_notif_remove_once_on_match): Likewise. (stop_reply_match_ptid_and_ws): Likewise. (kill_child_of_pending_fork): Likewise. (register_remote_g_packet_guess): Likewise. (remote_read_description_p): Likewise. (remote_read_description): Likewise. (free_actions_list_cleanup_wrapper): Likewise. (remote_async_serial_handler): Likewise. * rl78-tdep.c (rl78_get_opcode_byte): Likewise. (rl78_analyze_frame_prologue): Likewise. * rs6000-tdep.c (ppc_supply_gregset): Likewise. (ppc_supply_fpregset): Likewise. (ppc_supply_vsxregset): Likewise. (ppc_supply_vrregset): Likewise. (ppc_collect_gregset): Likewise. (ppc_collect_fpregset): Likewise. (ppc_collect_vsxregset): Likewise. (ppc_collect_vrregset): Likewise. (e500_move_ev_register): Likewise. (do_regcache_raw_write): Likewise. (rs6000_frame_cache): Likewise. (rs6000_epilogue_frame_cache): Likewise. (rs6000_gdbarch_init): Likewise. * rx-tdep.c (rx_get_opcode_byte): Likewise. (rx_analyze_frame_prologue): Likewise. (rx_frame_type): Likewise. (rx_frame_sniffer_common): Likewise. * s390-linux-tdep.c (s390_check_for_saved): Likewise. (s390_frame_unwind_cache): Likewise. (s390_stub_frame_unwind_cache): Likewise. (s390_sigtramp_frame_unwind_cache): Likewise. * score-tdep.c (score_make_prologue_cache): Likewise. * sentinel-frame.c (sentinel_frame_prev_register): Likewise. (sentinel_frame_prev_arch): Likewise. * ser-base.c (fd_event): Likewise. (push_event): Likewise. (ser_base_write): Likewise. * ser-pipe.c (pipe_close): Likewise. * serial.c (serial_write): Likewise. * sh-tdep.c (sh_frame_cache): Likewise. (sh_stub_this_id): Likewise. * sh64-tdep.c (sh64_frame_cache): Likewise. * solib-aix.c (get_solib_aix_inferior_data): Likewise. (library_list_start_library): Likewise. (library_list_start_list): Likewise. (solib_aix_free_library_list): Likewise. * solib-darwin.c (get_darwin_info): Likewise. * solib-dsbt.c (get_dsbt_info): Likewise. * solib-spu.c (append_ocl_sos): Likewise. * solib-svr4.c (svr4_pspace_data_cleanup): Likewise. (get_svr4_info): Likewise. (library_list_start_library): Likewise. (svr4_library_list_start_list): Likewise. (hash_probe_and_action): Likewise. (equal_probe_and_action): Likewise. (svr4_update_solib_event_breakpoint): Likewise. (set_solib_svr4_fetch_link_map_offsets): Likewise. (svr4_fetch_link_map_offsets): Likewise. (svr4_have_link_map_offsets): Likewise. * solib-target.c (library_list_start_segment): Likewise. (library_list_start_section): Likewise. (library_list_start_library): Likewise. (library_list_end_library): Likewise. (library_list_start_list): Likewise. (solib_target_free_library_list): Likewise. * solib.c (solib_ops): Likewise. (set_solib_ops): Likewise. * sparc-sol2-tdep.c (sparc32_sol2_sigtramp_frame_cache): Likewise. * sparc-tdep.c (sparc_frame_cache): Likewise. (sparc32_frame_cache): Likewise. (sparc32_supply_gregset): Likewise. (sparc32_collect_gregset): Likewise. (sparc32_supply_fpregset): Likewise. (sparc32_collect_fpregset): Likewise. * sparc64-sol2-tdep.c (sparc64_sol2_sigtramp_frame_cache): Likewise. * sparc64-tdep.c (sparc64_supply_gregset): Likewise. (sparc64_collect_gregset): Likewise. (sparc64_supply_fpregset): Likewise. (sparc64_collect_fpregset): Likewise. * sparc64fbsd-tdep.c (sparc64fbsd_sigtramp_frame_cache): Likewise. * sparc64nbsd-tdep.c (sparc64nbsd_sigcontext_frame_cache): Likewise. * sparc64obsd-tdep.c (sparc64obsd_frame_cache): Likewise. (sparc64obsd_trapframe_cache): Likewise. * sparcnbsd-tdep.c (sparc32nbsd_sigcontext_frame_cache): Likewise. * sparcobsd-tdep.c (sparc32obsd_sigtramp_frame_cache): Likewise. * spu-multiarch.c (spu_gdbarch): Likewise. * spu-tdep.c (spu_frame_unwind_cache): Likewise. (spu2ppu_prev_arch): Likewise. (spu2ppu_this_id): Likewise. (spu2ppu_prev_register): Likewise. (spu2ppu_dealloc_cache): Likewise. (spu_dis_asm_print_address): Likewise. (gdb_print_insn_spu): Likewise. (spu_get_overlay_table): Likewise. * stabsread.c (rs6000_builtin_type): Likewise. * stack.c (do_print_variable_and_value): Likewise. * stap-probe.c (get_stap_base_address_1): Likewise. * symfile-debug.c (debug_qf_has_symbols): Likewise. (debug_qf_find_last_source_symtab): Likewise. (debug_qf_forget_cached_source_info): Likewise. (debug_qf_map_symtabs_matching_filename): Likewise. (debug_qf_lookup_symbol): Likewise. (debug_qf_print_stats): Likewise. (debug_qf_dump): Likewise. (debug_qf_relocate): Likewise. (debug_qf_expand_symtabs_for_function): Likewise. (debug_qf_expand_all_symtabs): Likewise. (debug_qf_expand_symtabs_with_fullname): Likewise. (debug_qf_map_matching_symbols): Likewise. (debug_qf_expand_symtabs_matching): Likewise. (debug_qf_find_pc_sect_compunit_symtab): Likewise. (debug_qf_map_symbol_filenames): Likewise. (debug_sym_get_probes): Likewise. (debug_sym_new_init): Likewise. (debug_sym_init): Likewise. (debug_sym_read): Likewise. (debug_sym_read_psymbols): Likewise. (debug_sym_finish): Likewise. (debug_sym_offsets): Likewise. (debug_sym_read_linetable): Likewise. (debug_sym_relocate): Likewise. (uninstall_symfile_debug_logging): Likewise. * symfile-mem.c (symbol_file_add_from_memory_wrapper): Likewise. * symfile.c (place_section): Likewise. (add_section_size_callback): Likewise. (load_progress): Likewise. (load_section_callback): Likewise. (clear_memory_write_data): Likewise. (allocate_symtab): Likewise. * symmisc.c (maintenance_expand_file_matcher): Likewise. * symtab.c (lookup_symtab_callback): Likewise. (hash_demangled_name_entry): Likewise. (eq_demangled_name_entry): Likewise. (get_symbol_cache): Likewise. (symbol_cache_cleanup): Likewise. (set_symbol_cache_size): Likewise. (symbol_cache_flush): Likewise. (maintenance_print_symbol_cache): Likewise. (maintenance_print_symbol_cache_statistics): Likewise. (delete_filename_seen_cache): Likewise. (output_partial_symbol_filename): Likewise. (search_symbols_file_matches): Likewise. (search_symbols_name_matches): Likewise. (do_free_completion_list): Likewise. (maybe_add_partial_symtab_filename): Likewise. (get_main_info): Likewise. (main_info_cleanup): Likewise. * target-dcache.c (target_dcache_cleanup): Likewise. (target_dcache_init_p): Likewise. (target_dcache_invalidate): Likewise. (target_dcache_get): Likewise. (target_dcache_get_or_init): Likewise. * target-descriptions.c (target_find_description): Likewise. (tdesc_find_type): Likewise. (tdesc_data_cleanup): Likewise. (tdesc_find_arch_register): Likewise. (tdesc_register_name): Likewise. (tdesc_register_type): Likewise. (tdesc_register_reggroup_p): Likewise. (set_tdesc_pseudo_register_name): Likewise. (set_tdesc_pseudo_register_type): Likewise. (set_tdesc_pseudo_register_reggroup_p): Likewise. (tdesc_use_registers): Likewise. (free_target_description): Likewise. * target-memory.c (compare_block_starting_address): Likewise. (cleanup_request_data): Likewise. (cleanup_write_requests_vector): Likewise. * target.c (open_target): Likewise. (cleanup_restore_target_terminal): Likewise. (free_memory_read_result_vector): Likewise. * thread.c (disable_thread_stack_temporaries): Likewise. (finish_thread_state_cleanup): Likewise. (do_restore_current_thread_cleanup): Likewise. (restore_current_thread_cleanup_dtor): Likewise. (set_thread_refcount): Likewise. (tp_array_compar): Likewise. (do_captured_thread_select): Likewise. * tic6x-tdep.c (tic6x_frame_unwind_cache): Likewise. (tic6x_stub_this_id): Likewise. * tilegx-tdep.c (tilegx_frame_cache): Likewise. * top.c (do_restore_instream_cleanup): Likewise. (gdb_readline_wrapper_cleanup): Likewise. (kill_or_detach): Likewise. (print_inferior_quit_action): Likewise. * tracefile-tfile.c (match_blocktype): Likewise. (build_traceframe_info): Likewise. * tracefile.c (trace_file_writer_xfree): Likewise. * tracepoint.c (memrange_cmp): Likewise. (do_collect_symbol): Likewise. (do_clear_collection_list): Likewise. (do_restore_current_traceframe_cleanup): Likewise. (restore_current_traceframe_cleanup_dtor): Likewise. (free_current_marker): Likewise. (traceframe_info_start_memory): Likewise. (traceframe_info_start_tvar): Likewise. (free_result): Likewise. * tramp-frame.c (tramp_frame_cache): Likewise. * tui/tui-file.c (tui_file_delete): Likewise. (tui_fileopen): Likewise. (tui_sfileopen): Likewise. (tui_file_isatty): Likewise. (tui_file_rewind): Likewise. (tui_file_put): Likewise. (tui_file_fputs): Likewise. (tui_file_get_strbuf): Likewise. (tui_file_adjust_strbuf): Likewise. (tui_file_flush): Likewise. * tui/tui-layout.c (make_command_window): Likewise. (make_data_window): Likewise. (show_source_disasm_command): Likewise. (show_data): Likewise. (make_source_or_disasm_window): Likewise. (show_source_or_disasm_and_command): Likewise. * tui/tui-out.c (tui_field_int): Likewise. (tui_field_string): Likewise. (tui_field_fmt): Likewise. (tui_text): Likewise. * typeprint.c (hash_typedef_field): Likewise. (eq_typedef_field): Likewise. (do_free_typedef_hash): Likewise. (copy_typedef_hash_element): Likewise. (do_free_global_table): Likewise. (find_global_typedef): Likewise. (find_typedef_in_hash): Likewise. * ui-file.c (ui_file_write_for_put): Likewise. (do_ui_file_xstrdup): Likewise. (mem_file_delete): Likewise. (mem_file_rewind): Likewise. (mem_file_put): Likewise. (mem_file_write): Likewise. (stdio_file_delete): Likewise. (stdio_file_flush): Likewise. (stdio_file_read): Likewise. (stdio_file_write): Likewise. (stdio_file_write_async_safe): Likewise. (stdio_file_fputs): Likewise. (stdio_file_isatty): Likewise. (stdio_file_fseek): Likewise. (tee_file_delete): Likewise. (tee_file_flush): Likewise. (tee_file_write): Likewise. (tee_file_fputs): Likewise. (tee_file_isatty): Likewise. * ui-out.c (do_cleanup_table_end): Likewise. (do_cleanup_end): Likewise. * user-regs.c (user_reg_add): Likewise. (user_reg_map_name_to_regnum): Likewise. (usernum_to_user_reg): Likewise. (maintenance_print_user_registers): Likewise. * utils.c (do_bfd_close_cleanup): Likewise. (do_fclose_cleanup): Likewise. (do_obstack_free): Likewise. (do_ui_file_delete): Likewise. (do_ui_out_redirect_pop): Likewise. (do_free_section_addr_info): Likewise. (restore_integer): Likewise. (do_unpush_target): Likewise. (do_htab_delete_cleanup): Likewise. (do_restore_ui_file): Likewise. (do_value_free): Likewise. (do_free_so): Likewise. (free_current_contents): Likewise. (do_regfree_cleanup): Likewise. (core_addr_hash): Likewise. (core_addr_eq): Likewise. (do_free_char_ptr_vec): Likewise. * v850-tdep.c (v850_frame_cache): Likewise. * varobj.c (do_free_variable_cleanup): Likewise. * vax-tdep.c (vax_supply_gregset): Likewise. (vax_frame_cache): Likewise. * vaxobsd-tdep.c (vaxobsd_sigtramp_frame_cache): Likewise. * xml-support.c (gdb_xml_body_text): Likewise. (gdb_xml_values_cleanup): Likewise. (gdb_xml_start_element): Likewise. (gdb_xml_start_element_wrapper): Likewise. (gdb_xml_end_element): Likewise. (gdb_xml_end_element_wrapper): Likewise. (gdb_xml_cleanup): Likewise. (gdb_xml_fetch_external_entity): Likewise. (gdb_xml_parse_attr_enum): Likewise. (xinclude_start_include): Likewise. (xinclude_end_include): Likewise. (xml_xinclude_default): Likewise. (xml_xinclude_start_doctype): Likewise. (xml_xinclude_end_doctype): Likewise. (xml_xinclude_cleanup): Likewise. (xml_fetch_content_from_file): Likewise. * xml-syscall.c (free_syscalls_info): Likewise. (syscall_start_syscall): Likewise. * xml-tdesc.c (tdesc_end_arch): Likewise. (tdesc_end_osabi): Likewise. (tdesc_end_compatible): Likewise. (tdesc_start_target): Likewise. (tdesc_start_feature): Likewise. (tdesc_start_reg): Likewise. (tdesc_start_union): Likewise. (tdesc_start_struct): Likewise. (tdesc_start_flags): Likewise. (tdesc_start_field): Likewise. (tdesc_start_vector): Likewise. (fetch_available_features_from_target): Likewise. * xstormy16-tdep.c (xstormy16_frame_cache): Likewise. * xtensa-tdep.c (xtensa_supply_gregset): Likewise. (xtensa_frame_cache): Likewise. (xtensa_frame_prev_register): Likewise. (xtensa_extract_return_value): Likewise.
1742 lines
48 KiB
C
1742 lines
48 KiB
C
/* SystemTap probe support for GDB.
|
||
|
||
Copyright (C) 2012-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 "stap-probe.h"
|
||
#include "probe.h"
|
||
#include "vec.h"
|
||
#include "ui-out.h"
|
||
#include "objfiles.h"
|
||
#include "arch-utils.h"
|
||
#include "command.h"
|
||
#include "gdbcmd.h"
|
||
#include "filenames.h"
|
||
#include "value.h"
|
||
#include "ax.h"
|
||
#include "ax-gdb.h"
|
||
#include "complaints.h"
|
||
#include "cli/cli-utils.h"
|
||
#include "linespec.h"
|
||
#include "user-regs.h"
|
||
#include "parser-defs.h"
|
||
#include "language.h"
|
||
#include "elf-bfd.h"
|
||
|
||
#include <ctype.h>
|
||
|
||
/* The name of the SystemTap section where we will find information about
|
||
the probes. */
|
||
|
||
#define STAP_BASE_SECTION_NAME ".stapsdt.base"
|
||
|
||
/* Forward declaration. */
|
||
|
||
extern const struct probe_ops stap_probe_ops;
|
||
|
||
/* Should we display debug information for the probe's argument expression
|
||
parsing? */
|
||
|
||
static unsigned int stap_expression_debug = 0;
|
||
|
||
/* The various possibilities of bitness defined for a probe's argument.
|
||
|
||
The relationship is:
|
||
|
||
- STAP_ARG_BITNESS_UNDEFINED: The user hasn't specified the bitness.
|
||
- STAP_ARG_BITNESS_8BIT_UNSIGNED: argument string starts with `1@'.
|
||
- STAP_ARG_BITNESS_8BIT_SIGNED: argument string starts with `-1@'.
|
||
- STAP_ARG_BITNESS_16BIT_UNSIGNED: argument string starts with `2@'.
|
||
- STAP_ARG_BITNESS_16BIT_SIGNED: argument string starts with `-2@'.
|
||
- STAP_ARG_BITNESS_32BIT_UNSIGNED: argument string starts with `4@'.
|
||
- STAP_ARG_BITNESS_32BIT_SIGNED: argument string starts with `-4@'.
|
||
- STAP_ARG_BITNESS_64BIT_UNSIGNED: argument string starts with `8@'.
|
||
- STAP_ARG_BITNESS_64BIT_SIGNED: argument string starts with `-8@'. */
|
||
|
||
enum stap_arg_bitness
|
||
{
|
||
STAP_ARG_BITNESS_UNDEFINED,
|
||
STAP_ARG_BITNESS_8BIT_UNSIGNED,
|
||
STAP_ARG_BITNESS_8BIT_SIGNED,
|
||
STAP_ARG_BITNESS_16BIT_UNSIGNED,
|
||
STAP_ARG_BITNESS_16BIT_SIGNED,
|
||
STAP_ARG_BITNESS_32BIT_UNSIGNED,
|
||
STAP_ARG_BITNESS_32BIT_SIGNED,
|
||
STAP_ARG_BITNESS_64BIT_UNSIGNED,
|
||
STAP_ARG_BITNESS_64BIT_SIGNED,
|
||
};
|
||
|
||
/* The following structure represents a single argument for the probe. */
|
||
|
||
struct stap_probe_arg
|
||
{
|
||
/* The bitness of this argument. */
|
||
enum stap_arg_bitness bitness;
|
||
|
||
/* The corresponding `struct type *' to the bitness. */
|
||
struct type *atype;
|
||
|
||
/* The argument converted to an internal GDB expression. */
|
||
struct expression *aexpr;
|
||
};
|
||
|
||
typedef struct stap_probe_arg stap_probe_arg_s;
|
||
DEF_VEC_O (stap_probe_arg_s);
|
||
|
||
struct stap_probe
|
||
{
|
||
/* Generic information about the probe. This shall be the first element
|
||
of this struct, in order to maintain binary compatibility with the
|
||
`struct probe' and be able to fully abstract it. */
|
||
struct probe p;
|
||
|
||
/* If the probe has a semaphore associated, then this is the value of
|
||
it, relative to SECT_OFF_DATA. */
|
||
CORE_ADDR sem_addr;
|
||
|
||
/* One if the arguments have been parsed. */
|
||
unsigned int args_parsed : 1;
|
||
|
||
union
|
||
{
|
||
const char *text;
|
||
|
||
/* Information about each argument. This is an array of `stap_probe_arg',
|
||
with each entry representing one argument. */
|
||
VEC (stap_probe_arg_s) *vec;
|
||
}
|
||
args_u;
|
||
};
|
||
|
||
/* When parsing the arguments, we have to establish different precedences
|
||
for the various kinds of asm operators. This enumeration represents those
|
||
precedences.
|
||
|
||
This logic behind this is available at
|
||
<http://sourceware.org/binutils/docs/as/Infix-Ops.html#Infix-Ops>, or using
|
||
the command "info '(as)Infix Ops'". */
|
||
|
||
enum stap_operand_prec
|
||
{
|
||
/* Lowest precedence, used for non-recognized operands or for the beginning
|
||
of the parsing process. */
|
||
STAP_OPERAND_PREC_NONE = 0,
|
||
|
||
/* Precedence of logical OR. */
|
||
STAP_OPERAND_PREC_LOGICAL_OR,
|
||
|
||
/* Precedence of logical AND. */
|
||
STAP_OPERAND_PREC_LOGICAL_AND,
|
||
|
||
/* Precedence of additive (plus, minus) and comparative (equal, less,
|
||
greater-than, etc) operands. */
|
||
STAP_OPERAND_PREC_ADD_CMP,
|
||
|
||
/* Precedence of bitwise operands (bitwise OR, XOR, bitwise AND,
|
||
logical NOT). */
|
||
STAP_OPERAND_PREC_BITWISE,
|
||
|
||
/* Precedence of multiplicative operands (multiplication, division,
|
||
remainder, left shift and right shift). */
|
||
STAP_OPERAND_PREC_MUL
|
||
};
|
||
|
||
static void stap_parse_argument_1 (struct stap_parse_info *p, int has_lhs,
|
||
enum stap_operand_prec prec);
|
||
|
||
static void stap_parse_argument_conditionally (struct stap_parse_info *p);
|
||
|
||
/* Returns 1 if *S is an operator, zero otherwise. */
|
||
|
||
static int stap_is_operator (const char *op);
|
||
|
||
static void
|
||
show_stapexpressiondebug (struct ui_file *file, int from_tty,
|
||
struct cmd_list_element *c, const char *value)
|
||
{
|
||
fprintf_filtered (file, _("SystemTap Probe expression debugging is %s.\n"),
|
||
value);
|
||
}
|
||
|
||
/* Returns the operator precedence level of OP, or STAP_OPERAND_PREC_NONE
|
||
if the operator code was not recognized. */
|
||
|
||
static enum stap_operand_prec
|
||
stap_get_operator_prec (enum exp_opcode op)
|
||
{
|
||
switch (op)
|
||
{
|
||
case BINOP_LOGICAL_OR:
|
||
return STAP_OPERAND_PREC_LOGICAL_OR;
|
||
|
||
case BINOP_LOGICAL_AND:
|
||
return STAP_OPERAND_PREC_LOGICAL_AND;
|
||
|
||
case BINOP_ADD:
|
||
case BINOP_SUB:
|
||
case BINOP_EQUAL:
|
||
case BINOP_NOTEQUAL:
|
||
case BINOP_LESS:
|
||
case BINOP_LEQ:
|
||
case BINOP_GTR:
|
||
case BINOP_GEQ:
|
||
return STAP_OPERAND_PREC_ADD_CMP;
|
||
|
||
case BINOP_BITWISE_IOR:
|
||
case BINOP_BITWISE_AND:
|
||
case BINOP_BITWISE_XOR:
|
||
case UNOP_LOGICAL_NOT:
|
||
return STAP_OPERAND_PREC_BITWISE;
|
||
|
||
case BINOP_MUL:
|
||
case BINOP_DIV:
|
||
case BINOP_REM:
|
||
case BINOP_LSH:
|
||
case BINOP_RSH:
|
||
return STAP_OPERAND_PREC_MUL;
|
||
|
||
default:
|
||
return STAP_OPERAND_PREC_NONE;
|
||
}
|
||
}
|
||
|
||
/* Given S, read the operator in it and fills the OP pointer with its code.
|
||
Return 1 on success, zero if the operator was not recognized. */
|
||
|
||
static enum exp_opcode
|
||
stap_get_opcode (const char **s)
|
||
{
|
||
const char c = **s;
|
||
enum exp_opcode op;
|
||
|
||
*s += 1;
|
||
|
||
switch (c)
|
||
{
|
||
case '*':
|
||
op = BINOP_MUL;
|
||
break;
|
||
|
||
case '/':
|
||
op = BINOP_DIV;
|
||
break;
|
||
|
||
case '%':
|
||
op = BINOP_REM;
|
||
break;
|
||
|
||
case '<':
|
||
op = BINOP_LESS;
|
||
if (**s == '<')
|
||
{
|
||
*s += 1;
|
||
op = BINOP_LSH;
|
||
}
|
||
else if (**s == '=')
|
||
{
|
||
*s += 1;
|
||
op = BINOP_LEQ;
|
||
}
|
||
else if (**s == '>')
|
||
{
|
||
*s += 1;
|
||
op = BINOP_NOTEQUAL;
|
||
}
|
||
break;
|
||
|
||
case '>':
|
||
op = BINOP_GTR;
|
||
if (**s == '>')
|
||
{
|
||
*s += 1;
|
||
op = BINOP_RSH;
|
||
}
|
||
else if (**s == '=')
|
||
{
|
||
*s += 1;
|
||
op = BINOP_GEQ;
|
||
}
|
||
break;
|
||
|
||
case '|':
|
||
op = BINOP_BITWISE_IOR;
|
||
if (**s == '|')
|
||
{
|
||
*s += 1;
|
||
op = BINOP_LOGICAL_OR;
|
||
}
|
||
break;
|
||
|
||
case '&':
|
||
op = BINOP_BITWISE_AND;
|
||
if (**s == '&')
|
||
{
|
||
*s += 1;
|
||
op = BINOP_LOGICAL_AND;
|
||
}
|
||
break;
|
||
|
||
case '^':
|
||
op = BINOP_BITWISE_XOR;
|
||
break;
|
||
|
||
case '!':
|
||
op = UNOP_LOGICAL_NOT;
|
||
break;
|
||
|
||
case '+':
|
||
op = BINOP_ADD;
|
||
break;
|
||
|
||
case '-':
|
||
op = BINOP_SUB;
|
||
break;
|
||
|
||
case '=':
|
||
gdb_assert (**s == '=');
|
||
op = BINOP_EQUAL;
|
||
break;
|
||
|
||
default:
|
||
error (_("Invalid opcode in expression `%s' for SystemTap"
|
||
"probe"), *s);
|
||
}
|
||
|
||
return op;
|
||
}
|
||
|
||
/* Given the bitness of the argument, represented by B, return the
|
||
corresponding `struct type *'. */
|
||
|
||
static struct type *
|
||
stap_get_expected_argument_type (struct gdbarch *gdbarch,
|
||
enum stap_arg_bitness b,
|
||
const struct stap_probe *probe)
|
||
{
|
||
switch (b)
|
||
{
|
||
case STAP_ARG_BITNESS_UNDEFINED:
|
||
if (gdbarch_addr_bit (gdbarch) == 32)
|
||
return builtin_type (gdbarch)->builtin_uint32;
|
||
else
|
||
return builtin_type (gdbarch)->builtin_uint64;
|
||
|
||
case STAP_ARG_BITNESS_8BIT_UNSIGNED:
|
||
return builtin_type (gdbarch)->builtin_uint8;
|
||
|
||
case STAP_ARG_BITNESS_8BIT_SIGNED:
|
||
return builtin_type (gdbarch)->builtin_int8;
|
||
|
||
case STAP_ARG_BITNESS_16BIT_UNSIGNED:
|
||
return builtin_type (gdbarch)->builtin_uint16;
|
||
|
||
case STAP_ARG_BITNESS_16BIT_SIGNED:
|
||
return builtin_type (gdbarch)->builtin_int16;
|
||
|
||
case STAP_ARG_BITNESS_32BIT_SIGNED:
|
||
return builtin_type (gdbarch)->builtin_int32;
|
||
|
||
case STAP_ARG_BITNESS_32BIT_UNSIGNED:
|
||
return builtin_type (gdbarch)->builtin_uint32;
|
||
|
||
case STAP_ARG_BITNESS_64BIT_SIGNED:
|
||
return builtin_type (gdbarch)->builtin_int64;
|
||
|
||
case STAP_ARG_BITNESS_64BIT_UNSIGNED:
|
||
return builtin_type (gdbarch)->builtin_uint64;
|
||
|
||
default:
|
||
error (_("Undefined bitness for probe '%s'."),
|
||
probe->p.name);
|
||
break;
|
||
}
|
||
}
|
||
|
||
/* Helper function to check for a generic list of prefixes. GDBARCH
|
||
is the current gdbarch being used. S is the expression being
|
||
analyzed. If R is not NULL, it will be used to return the found
|
||
prefix. PREFIXES is the list of expected prefixes.
|
||
|
||
This function does a case-insensitive match.
|
||
|
||
Return 1 if any prefix has been found, zero otherwise. */
|
||
|
||
static int
|
||
stap_is_generic_prefix (struct gdbarch *gdbarch, const char *s,
|
||
const char **r, const char *const *prefixes)
|
||
{
|
||
const char *const *p;
|
||
|
||
if (prefixes == NULL)
|
||
{
|
||
if (r != NULL)
|
||
*r = "";
|
||
|
||
return 1;
|
||
}
|
||
|
||
for (p = prefixes; *p != NULL; ++p)
|
||
if (strncasecmp (s, *p, strlen (*p)) == 0)
|
||
{
|
||
if (r != NULL)
|
||
*r = *p;
|
||
|
||
return 1;
|
||
}
|
||
|
||
return 0;
|
||
}
|
||
|
||
/* Return 1 if S points to a register prefix, zero otherwise. For a
|
||
description of the arguments, look at stap_is_generic_prefix. */
|
||
|
||
static int
|
||
stap_is_register_prefix (struct gdbarch *gdbarch, const char *s,
|
||
const char **r)
|
||
{
|
||
const char *const *t = gdbarch_stap_register_prefixes (gdbarch);
|
||
|
||
return stap_is_generic_prefix (gdbarch, s, r, t);
|
||
}
|
||
|
||
/* Return 1 if S points to a register indirection prefix, zero
|
||
otherwise. For a description of the arguments, look at
|
||
stap_is_generic_prefix. */
|
||
|
||
static int
|
||
stap_is_register_indirection_prefix (struct gdbarch *gdbarch, const char *s,
|
||
const char **r)
|
||
{
|
||
const char *const *t = gdbarch_stap_register_indirection_prefixes (gdbarch);
|
||
|
||
return stap_is_generic_prefix (gdbarch, s, r, t);
|
||
}
|
||
|
||
/* Return 1 if S points to an integer prefix, zero otherwise. For a
|
||
description of the arguments, look at stap_is_generic_prefix.
|
||
|
||
This function takes care of analyzing whether we are dealing with
|
||
an expected integer prefix, or, if there is no integer prefix to be
|
||
expected, whether we are dealing with a digit. It does a
|
||
case-insensitive match. */
|
||
|
||
static int
|
||
stap_is_integer_prefix (struct gdbarch *gdbarch, const char *s,
|
||
const char **r)
|
||
{
|
||
const char *const *t = gdbarch_stap_integer_prefixes (gdbarch);
|
||
const char *const *p;
|
||
|
||
if (t == NULL)
|
||
{
|
||
/* A NULL value here means that integers do not have a prefix.
|
||
We just check for a digit then. */
|
||
if (r != NULL)
|
||
*r = "";
|
||
|
||
return isdigit (*s);
|
||
}
|
||
|
||
for (p = t; *p != NULL; ++p)
|
||
{
|
||
size_t len = strlen (*p);
|
||
|
||
if ((len == 0 && isdigit (*s))
|
||
|| (len > 0 && strncasecmp (s, *p, len) == 0))
|
||
{
|
||
/* Integers may or may not have a prefix. The "len == 0"
|
||
check covers the case when integers do not have a prefix
|
||
(therefore, we just check if we have a digit). The call
|
||
to "strncasecmp" covers the case when they have a
|
||
prefix. */
|
||
if (r != NULL)
|
||
*r = *p;
|
||
|
||
return 1;
|
||
}
|
||
}
|
||
|
||
return 0;
|
||
}
|
||
|
||
/* Helper function to check for a generic list of suffixes. If we are
|
||
not expecting any suffixes, then it just returns 1. If we are
|
||
expecting at least one suffix, then it returns 1 if a suffix has
|
||
been found, zero otherwise. GDBARCH is the current gdbarch being
|
||
used. S is the expression being analyzed. If R is not NULL, it
|
||
will be used to return the found suffix. SUFFIXES is the list of
|
||
expected suffixes. This function does a case-insensitive
|
||
match. */
|
||
|
||
static int
|
||
stap_generic_check_suffix (struct gdbarch *gdbarch, const char *s,
|
||
const char **r, const char *const *suffixes)
|
||
{
|
||
const char *const *p;
|
||
int found = 0;
|
||
|
||
if (suffixes == NULL)
|
||
{
|
||
if (r != NULL)
|
||
*r = "";
|
||
|
||
return 1;
|
||
}
|
||
|
||
for (p = suffixes; *p != NULL; ++p)
|
||
if (strncasecmp (s, *p, strlen (*p)) == 0)
|
||
{
|
||
if (r != NULL)
|
||
*r = *p;
|
||
|
||
found = 1;
|
||
break;
|
||
}
|
||
|
||
return found;
|
||
}
|
||
|
||
/* Return 1 if S points to an integer suffix, zero otherwise. For a
|
||
description of the arguments, look at
|
||
stap_generic_check_suffix. */
|
||
|
||
static int
|
||
stap_check_integer_suffix (struct gdbarch *gdbarch, const char *s,
|
||
const char **r)
|
||
{
|
||
const char *const *p = gdbarch_stap_integer_suffixes (gdbarch);
|
||
|
||
return stap_generic_check_suffix (gdbarch, s, r, p);
|
||
}
|
||
|
||
/* Return 1 if S points to a register suffix, zero otherwise. For a
|
||
description of the arguments, look at
|
||
stap_generic_check_suffix. */
|
||
|
||
static int
|
||
stap_check_register_suffix (struct gdbarch *gdbarch, const char *s,
|
||
const char **r)
|
||
{
|
||
const char *const *p = gdbarch_stap_register_suffixes (gdbarch);
|
||
|
||
return stap_generic_check_suffix (gdbarch, s, r, p);
|
||
}
|
||
|
||
/* Return 1 if S points to a register indirection suffix, zero
|
||
otherwise. For a description of the arguments, look at
|
||
stap_generic_check_suffix. */
|
||
|
||
static int
|
||
stap_check_register_indirection_suffix (struct gdbarch *gdbarch, const char *s,
|
||
const char **r)
|
||
{
|
||
const char *const *p = gdbarch_stap_register_indirection_suffixes (gdbarch);
|
||
|
||
return stap_generic_check_suffix (gdbarch, s, r, p);
|
||
}
|
||
|
||
/* Function responsible for parsing a register operand according to
|
||
SystemTap parlance. Assuming:
|
||
|
||
RP = register prefix
|
||
RS = register suffix
|
||
RIP = register indirection prefix
|
||
RIS = register indirection suffix
|
||
|
||
Then a register operand can be:
|
||
|
||
[RIP] [RP] REGISTER [RS] [RIS]
|
||
|
||
This function takes care of a register's indirection, displacement and
|
||
direct access. It also takes into consideration the fact that some
|
||
registers are named differently inside and outside GDB, e.g., PPC's
|
||
general-purpose registers are represented by integers in the assembly
|
||
language (e.g., `15' is the 15th general-purpose register), but inside
|
||
GDB they have a prefix (the letter `r') appended. */
|
||
|
||
static void
|
||
stap_parse_register_operand (struct stap_parse_info *p)
|
||
{
|
||
/* Simple flag to indicate whether we have seen a minus signal before
|
||
certain number. */
|
||
int got_minus = 0;
|
||
/* Flags to indicate whether this register access is being displaced and/or
|
||
indirected. */
|
||
int disp_p = 0, indirect_p = 0;
|
||
struct gdbarch *gdbarch = p->gdbarch;
|
||
/* Needed to generate the register name as a part of an expression. */
|
||
struct stoken str;
|
||
/* Variables used to extract the register name from the probe's
|
||
argument. */
|
||
const char *start;
|
||
char *regname;
|
||
int len;
|
||
const char *gdb_reg_prefix = gdbarch_stap_gdb_register_prefix (gdbarch);
|
||
int gdb_reg_prefix_len = gdb_reg_prefix ? strlen (gdb_reg_prefix) : 0;
|
||
const char *gdb_reg_suffix = gdbarch_stap_gdb_register_suffix (gdbarch);
|
||
int gdb_reg_suffix_len = gdb_reg_suffix ? strlen (gdb_reg_suffix) : 0;
|
||
const char *reg_prefix;
|
||
const char *reg_ind_prefix;
|
||
const char *reg_suffix;
|
||
const char *reg_ind_suffix;
|
||
|
||
/* Checking for a displacement argument. */
|
||
if (*p->arg == '+')
|
||
{
|
||
/* If it's a plus sign, we don't need to do anything, just advance the
|
||
pointer. */
|
||
++p->arg;
|
||
}
|
||
|
||
if (*p->arg == '-')
|
||
{
|
||
got_minus = 1;
|
||
++p->arg;
|
||
}
|
||
|
||
if (isdigit (*p->arg))
|
||
{
|
||
/* The value of the displacement. */
|
||
long displacement;
|
||
char *endp;
|
||
|
||
disp_p = 1;
|
||
displacement = strtol (p->arg, &endp, 10);
|
||
p->arg = endp;
|
||
|
||
/* Generating the expression for the displacement. */
|
||
write_exp_elt_opcode (&p->pstate, OP_LONG);
|
||
write_exp_elt_type (&p->pstate, builtin_type (gdbarch)->builtin_long);
|
||
write_exp_elt_longcst (&p->pstate, displacement);
|
||
write_exp_elt_opcode (&p->pstate, OP_LONG);
|
||
if (got_minus)
|
||
write_exp_elt_opcode (&p->pstate, UNOP_NEG);
|
||
}
|
||
|
||
/* Getting rid of register indirection prefix. */
|
||
if (stap_is_register_indirection_prefix (gdbarch, p->arg, ®_ind_prefix))
|
||
{
|
||
indirect_p = 1;
|
||
p->arg += strlen (reg_ind_prefix);
|
||
}
|
||
|
||
if (disp_p && !indirect_p)
|
||
error (_("Invalid register displacement syntax on expression `%s'."),
|
||
p->saved_arg);
|
||
|
||
/* Getting rid of register prefix. */
|
||
if (stap_is_register_prefix (gdbarch, p->arg, ®_prefix))
|
||
p->arg += strlen (reg_prefix);
|
||
|
||
/* Now we should have only the register name. Let's extract it and get
|
||
the associated number. */
|
||
start = p->arg;
|
||
|
||
/* We assume the register name is composed by letters and numbers. */
|
||
while (isalnum (*p->arg))
|
||
++p->arg;
|
||
|
||
len = p->arg - start;
|
||
|
||
regname = (char *) alloca (len + gdb_reg_prefix_len + gdb_reg_suffix_len + 1);
|
||
regname[0] = '\0';
|
||
|
||
/* We only add the GDB's register prefix/suffix if we are dealing with
|
||
a numeric register. */
|
||
if (gdb_reg_prefix && isdigit (*start))
|
||
{
|
||
strncpy (regname, gdb_reg_prefix, gdb_reg_prefix_len);
|
||
strncpy (regname + gdb_reg_prefix_len, start, len);
|
||
|
||
if (gdb_reg_suffix)
|
||
strncpy (regname + gdb_reg_prefix_len + len,
|
||
gdb_reg_suffix, gdb_reg_suffix_len);
|
||
|
||
len += gdb_reg_prefix_len + gdb_reg_suffix_len;
|
||
}
|
||
else
|
||
strncpy (regname, start, len);
|
||
|
||
regname[len] = '\0';
|
||
|
||
/* Is this a valid register name? */
|
||
if (user_reg_map_name_to_regnum (gdbarch, regname, len) == -1)
|
||
error (_("Invalid register name `%s' on expression `%s'."),
|
||
regname, p->saved_arg);
|
||
|
||
write_exp_elt_opcode (&p->pstate, OP_REGISTER);
|
||
str.ptr = regname;
|
||
str.length = len;
|
||
write_exp_string (&p->pstate, str);
|
||
write_exp_elt_opcode (&p->pstate, OP_REGISTER);
|
||
|
||
if (indirect_p)
|
||
{
|
||
if (disp_p)
|
||
write_exp_elt_opcode (&p->pstate, BINOP_ADD);
|
||
|
||
/* Casting to the expected type. */
|
||
write_exp_elt_opcode (&p->pstate, UNOP_CAST);
|
||
write_exp_elt_type (&p->pstate, lookup_pointer_type (p->arg_type));
|
||
write_exp_elt_opcode (&p->pstate, UNOP_CAST);
|
||
|
||
write_exp_elt_opcode (&p->pstate, UNOP_IND);
|
||
}
|
||
|
||
/* Getting rid of the register name suffix. */
|
||
if (stap_check_register_suffix (gdbarch, p->arg, ®_suffix))
|
||
p->arg += strlen (reg_suffix);
|
||
else
|
||
error (_("Missing register name suffix on expression `%s'."),
|
||
p->saved_arg);
|
||
|
||
/* Getting rid of the register indirection suffix. */
|
||
if (indirect_p)
|
||
{
|
||
if (stap_check_register_indirection_suffix (gdbarch, p->arg,
|
||
®_ind_suffix))
|
||
p->arg += strlen (reg_ind_suffix);
|
||
else
|
||
error (_("Missing indirection suffix on expression `%s'."),
|
||
p->saved_arg);
|
||
}
|
||
}
|
||
|
||
/* This function is responsible for parsing a single operand.
|
||
|
||
A single operand can be:
|
||
|
||
- an unary operation (e.g., `-5', `~2', or even with subexpressions
|
||
like `-(2 + 1)')
|
||
- a register displacement, which will be treated as a register
|
||
operand (e.g., `-4(%eax)' on x86)
|
||
- a numeric constant, or
|
||
- a register operand (see function `stap_parse_register_operand')
|
||
|
||
The function also calls special-handling functions to deal with
|
||
unrecognized operands, allowing arch-specific parsers to be
|
||
created. */
|
||
|
||
static void
|
||
stap_parse_single_operand (struct stap_parse_info *p)
|
||
{
|
||
struct gdbarch *gdbarch = p->gdbarch;
|
||
const char *int_prefix = NULL;
|
||
|
||
/* We first try to parse this token as a "special token". */
|
||
if (gdbarch_stap_parse_special_token_p (gdbarch))
|
||
if (gdbarch_stap_parse_special_token (gdbarch, p) != 0)
|
||
{
|
||
/* If the return value of the above function is not zero,
|
||
it means it successfully parsed the special token.
|
||
|
||
If it is NULL, we try to parse it using our method. */
|
||
return;
|
||
}
|
||
|
||
if (*p->arg == '-' || *p->arg == '~' || *p->arg == '+')
|
||
{
|
||
char c = *p->arg;
|
||
/* We use this variable to do a lookahead. */
|
||
const char *tmp = p->arg;
|
||
int has_digit = 0;
|
||
|
||
/* Skipping signal. */
|
||
++tmp;
|
||
|
||
/* This is an unary operation. Here is a list of allowed tokens
|
||
here:
|
||
|
||
- numeric literal;
|
||
- number (from register displacement)
|
||
- subexpression (beginning with `(')
|
||
|
||
We handle the register displacement here, and the other cases
|
||
recursively. */
|
||
if (p->inside_paren_p)
|
||
tmp = skip_spaces_const (tmp);
|
||
|
||
while (isdigit (*tmp))
|
||
{
|
||
/* We skip the digit here because we are only interested in
|
||
knowing what kind of unary operation this is. The digit
|
||
will be handled by one of the functions that will be
|
||
called below ('stap_parse_argument_conditionally' or
|
||
'stap_parse_register_operand'). */
|
||
++tmp;
|
||
has_digit = 1;
|
||
}
|
||
|
||
if (has_digit && stap_is_register_indirection_prefix (gdbarch, tmp,
|
||
NULL))
|
||
{
|
||
/* If we are here, it means it is a displacement. The only
|
||
operations allowed here are `-' and `+'. */
|
||
if (c == '~')
|
||
error (_("Invalid operator `%c' for register displacement "
|
||
"on expression `%s'."), c, p->saved_arg);
|
||
|
||
stap_parse_register_operand (p);
|
||
}
|
||
else
|
||
{
|
||
/* This is not a displacement. We skip the operator, and
|
||
deal with it when the recursion returns. */
|
||
++p->arg;
|
||
stap_parse_argument_conditionally (p);
|
||
if (c == '-')
|
||
write_exp_elt_opcode (&p->pstate, UNOP_NEG);
|
||
else if (c == '~')
|
||
write_exp_elt_opcode (&p->pstate, UNOP_COMPLEMENT);
|
||
}
|
||
}
|
||
else if (isdigit (*p->arg))
|
||
{
|
||
/* A temporary variable, needed for lookahead. */
|
||
const char *tmp = p->arg;
|
||
char *endp;
|
||
long number;
|
||
|
||
/* We can be dealing with a numeric constant, or with a register
|
||
displacement. */
|
||
number = strtol (tmp, &endp, 10);
|
||
tmp = endp;
|
||
|
||
if (p->inside_paren_p)
|
||
tmp = skip_spaces_const (tmp);
|
||
|
||
/* If "stap_is_integer_prefix" returns true, it means we can
|
||
accept integers without a prefix here. But we also need to
|
||
check whether the next token (i.e., "tmp") is not a register
|
||
indirection prefix. */
|
||
if (stap_is_integer_prefix (gdbarch, p->arg, NULL)
|
||
&& !stap_is_register_indirection_prefix (gdbarch, tmp, NULL))
|
||
{
|
||
const char *int_suffix;
|
||
|
||
/* We are dealing with a numeric constant. */
|
||
write_exp_elt_opcode (&p->pstate, OP_LONG);
|
||
write_exp_elt_type (&p->pstate,
|
||
builtin_type (gdbarch)->builtin_long);
|
||
write_exp_elt_longcst (&p->pstate, number);
|
||
write_exp_elt_opcode (&p->pstate, OP_LONG);
|
||
|
||
p->arg = tmp;
|
||
|
||
if (stap_check_integer_suffix (gdbarch, p->arg, &int_suffix))
|
||
p->arg += strlen (int_suffix);
|
||
else
|
||
error (_("Invalid constant suffix on expression `%s'."),
|
||
p->saved_arg);
|
||
}
|
||
else if (stap_is_register_indirection_prefix (gdbarch, tmp, NULL))
|
||
stap_parse_register_operand (p);
|
||
else
|
||
error (_("Unknown numeric token on expression `%s'."),
|
||
p->saved_arg);
|
||
}
|
||
else if (stap_is_integer_prefix (gdbarch, p->arg, &int_prefix))
|
||
{
|
||
/* We are dealing with a numeric constant. */
|
||
long number;
|
||
char *endp;
|
||
const char *int_suffix;
|
||
|
||
p->arg += strlen (int_prefix);
|
||
number = strtol (p->arg, &endp, 10);
|
||
p->arg = endp;
|
||
|
||
write_exp_elt_opcode (&p->pstate, OP_LONG);
|
||
write_exp_elt_type (&p->pstate, builtin_type (gdbarch)->builtin_long);
|
||
write_exp_elt_longcst (&p->pstate, number);
|
||
write_exp_elt_opcode (&p->pstate, OP_LONG);
|
||
|
||
if (stap_check_integer_suffix (gdbarch, p->arg, &int_suffix))
|
||
p->arg += strlen (int_suffix);
|
||
else
|
||
error (_("Invalid constant suffix on expression `%s'."),
|
||
p->saved_arg);
|
||
}
|
||
else if (stap_is_register_prefix (gdbarch, p->arg, NULL)
|
||
|| stap_is_register_indirection_prefix (gdbarch, p->arg, NULL))
|
||
stap_parse_register_operand (p);
|
||
else
|
||
error (_("Operator `%c' not recognized on expression `%s'."),
|
||
*p->arg, p->saved_arg);
|
||
}
|
||
|
||
/* This function parses an argument conditionally, based on single or
|
||
non-single operands. A non-single operand would be a parenthesized
|
||
expression (e.g., `(2 + 1)'), and a single operand is anything that
|
||
starts with `-', `~', `+' (i.e., unary operators), a digit, or
|
||
something recognized by `gdbarch_stap_is_single_operand'. */
|
||
|
||
static void
|
||
stap_parse_argument_conditionally (struct stap_parse_info *p)
|
||
{
|
||
gdb_assert (gdbarch_stap_is_single_operand_p (p->gdbarch));
|
||
|
||
if (*p->arg == '-' || *p->arg == '~' || *p->arg == '+' /* Unary. */
|
||
|| isdigit (*p->arg)
|
||
|| gdbarch_stap_is_single_operand (p->gdbarch, p->arg))
|
||
stap_parse_single_operand (p);
|
||
else if (*p->arg == '(')
|
||
{
|
||
/* We are dealing with a parenthesized operand. It means we
|
||
have to parse it as it was a separate expression, without
|
||
left-side or precedence. */
|
||
++p->arg;
|
||
p->arg = skip_spaces_const (p->arg);
|
||
++p->inside_paren_p;
|
||
|
||
stap_parse_argument_1 (p, 0, STAP_OPERAND_PREC_NONE);
|
||
|
||
--p->inside_paren_p;
|
||
if (*p->arg != ')')
|
||
error (_("Missign close-paren on expression `%s'."),
|
||
p->saved_arg);
|
||
|
||
++p->arg;
|
||
if (p->inside_paren_p)
|
||
p->arg = skip_spaces_const (p->arg);
|
||
}
|
||
else
|
||
error (_("Cannot parse expression `%s'."), p->saved_arg);
|
||
}
|
||
|
||
/* Helper function for `stap_parse_argument'. Please, see its comments to
|
||
better understand what this function does. */
|
||
|
||
static void
|
||
stap_parse_argument_1 (struct stap_parse_info *p, int has_lhs,
|
||
enum stap_operand_prec prec)
|
||
{
|
||
/* This is an operator-precedence parser.
|
||
|
||
We work with left- and right-sides of expressions, and
|
||
parse them depending on the precedence of the operators
|
||
we find. */
|
||
|
||
gdb_assert (p->arg != NULL);
|
||
|
||
if (p->inside_paren_p)
|
||
p->arg = skip_spaces_const (p->arg);
|
||
|
||
if (!has_lhs)
|
||
{
|
||
/* We were called without a left-side, either because this is the
|
||
first call, or because we were called to parse a parenthesized
|
||
expression. It doesn't really matter; we have to parse the
|
||
left-side in order to continue the process. */
|
||
stap_parse_argument_conditionally (p);
|
||
}
|
||
|
||
/* Start to parse the right-side, and to "join" left and right sides
|
||
depending on the operation specified.
|
||
|
||
This loop shall continue until we run out of characters in the input,
|
||
or until we find a close-parenthesis, which means that we've reached
|
||
the end of a sub-expression. */
|
||
while (*p->arg != '\0' && *p->arg != ')' && !isspace (*p->arg))
|
||
{
|
||
const char *tmp_exp_buf;
|
||
enum exp_opcode opcode;
|
||
enum stap_operand_prec cur_prec;
|
||
|
||
if (!stap_is_operator (p->arg))
|
||
error (_("Invalid operator `%c' on expression `%s'."), *p->arg,
|
||
p->saved_arg);
|
||
|
||
/* We have to save the current value of the expression buffer because
|
||
the `stap_get_opcode' modifies it in order to get the current
|
||
operator. If this operator's precedence is lower than PREC, we
|
||
should return and not advance the expression buffer pointer. */
|
||
tmp_exp_buf = p->arg;
|
||
opcode = stap_get_opcode (&tmp_exp_buf);
|
||
|
||
cur_prec = stap_get_operator_prec (opcode);
|
||
if (cur_prec < prec)
|
||
{
|
||
/* If the precedence of the operator that we are seeing now is
|
||
lower than the precedence of the first operator seen before
|
||
this parsing process began, it means we should stop parsing
|
||
and return. */
|
||
break;
|
||
}
|
||
|
||
p->arg = tmp_exp_buf;
|
||
if (p->inside_paren_p)
|
||
p->arg = skip_spaces_const (p->arg);
|
||
|
||
/* Parse the right-side of the expression. */
|
||
stap_parse_argument_conditionally (p);
|
||
|
||
/* While we still have operators, try to parse another
|
||
right-side, but using the current right-side as a left-side. */
|
||
while (*p->arg != '\0' && stap_is_operator (p->arg))
|
||
{
|
||
enum exp_opcode lookahead_opcode;
|
||
enum stap_operand_prec lookahead_prec;
|
||
|
||
/* Saving the current expression buffer position. The explanation
|
||
is the same as above. */
|
||
tmp_exp_buf = p->arg;
|
||
lookahead_opcode = stap_get_opcode (&tmp_exp_buf);
|
||
lookahead_prec = stap_get_operator_prec (lookahead_opcode);
|
||
|
||
if (lookahead_prec <= prec)
|
||
{
|
||
/* If we are dealing with an operator whose precedence is lower
|
||
than the first one, just abandon the attempt. */
|
||
break;
|
||
}
|
||
|
||
/* Parse the right-side of the expression, but since we already
|
||
have a left-side at this point, set `has_lhs' to 1. */
|
||
stap_parse_argument_1 (p, 1, lookahead_prec);
|
||
}
|
||
|
||
write_exp_elt_opcode (&p->pstate, opcode);
|
||
}
|
||
}
|
||
|
||
/* Parse a probe's argument.
|
||
|
||
Assuming that:
|
||
|
||
LP = literal integer prefix
|
||
LS = literal integer suffix
|
||
|
||
RP = register prefix
|
||
RS = register suffix
|
||
|
||
RIP = register indirection prefix
|
||
RIS = register indirection suffix
|
||
|
||
This routine assumes that arguments' tokens are of the form:
|
||
|
||
- [LP] NUMBER [LS]
|
||
- [RP] REGISTER [RS]
|
||
- [RIP] [RP] REGISTER [RS] [RIS]
|
||
- If we find a number without LP, we try to parse it as a literal integer
|
||
constant (if LP == NULL), or as a register displacement.
|
||
- We count parenthesis, and only skip whitespaces if we are inside them.
|
||
- If we find an operator, we skip it.
|
||
|
||
This function can also call a special function that will try to match
|
||
unknown tokens. It will return 1 if the argument has been parsed
|
||
successfully, or zero otherwise. */
|
||
|
||
static struct expression *
|
||
stap_parse_argument (const char **arg, struct type *atype,
|
||
struct gdbarch *gdbarch)
|
||
{
|
||
struct stap_parse_info p;
|
||
struct cleanup *back_to;
|
||
|
||
/* We need to initialize the expression buffer, in order to begin
|
||
our parsing efforts. We use language_c here because we may need
|
||
to do pointer arithmetics. */
|
||
initialize_expout (&p.pstate, 10, language_def (language_c), gdbarch);
|
||
back_to = make_cleanup (free_current_contents, &p.pstate.expout);
|
||
|
||
p.saved_arg = *arg;
|
||
p.arg = *arg;
|
||
p.arg_type = atype;
|
||
p.gdbarch = gdbarch;
|
||
p.inside_paren_p = 0;
|
||
|
||
stap_parse_argument_1 (&p, 0, STAP_OPERAND_PREC_NONE);
|
||
|
||
discard_cleanups (back_to);
|
||
|
||
gdb_assert (p.inside_paren_p == 0);
|
||
|
||
/* Casting the final expression to the appropriate type. */
|
||
write_exp_elt_opcode (&p.pstate, UNOP_CAST);
|
||
write_exp_elt_type (&p.pstate, atype);
|
||
write_exp_elt_opcode (&p.pstate, UNOP_CAST);
|
||
|
||
reallocate_expout (&p.pstate);
|
||
|
||
p.arg = skip_spaces_const (p.arg);
|
||
*arg = p.arg;
|
||
|
||
/* We can safely return EXPOUT here. */
|
||
return p.pstate.expout;
|
||
}
|
||
|
||
/* Function which parses an argument string from PROBE, correctly splitting
|
||
the arguments and storing their information in properly ways.
|
||
|
||
Consider the following argument string (x86 syntax):
|
||
|
||
`4@%eax 4@$10'
|
||
|
||
We have two arguments, `%eax' and `$10', both with 32-bit unsigned bitness.
|
||
This function basically handles them, properly filling some structures with
|
||
this information. */
|
||
|
||
static void
|
||
stap_parse_probe_arguments (struct stap_probe *probe, struct gdbarch *gdbarch)
|
||
{
|
||
const char *cur;
|
||
|
||
gdb_assert (!probe->args_parsed);
|
||
cur = probe->args_u.text;
|
||
probe->args_parsed = 1;
|
||
probe->args_u.vec = NULL;
|
||
|
||
if (cur == NULL || *cur == '\0' || *cur == ':')
|
||
return;
|
||
|
||
while (*cur != '\0')
|
||
{
|
||
struct stap_probe_arg arg;
|
||
enum stap_arg_bitness b;
|
||
int got_minus = 0;
|
||
struct expression *expr;
|
||
|
||
memset (&arg, 0, sizeof (arg));
|
||
|
||
/* We expect to find something like:
|
||
|
||
N@OP
|
||
|
||
Where `N' can be [+,-][1,2,4,8]. This is not mandatory, so
|
||
we check it here. If we don't find it, go to the next
|
||
state. */
|
||
if ((cur[0] == '-' && isdigit (cur[1]) && cur[2] == '@')
|
||
|| (isdigit (cur[0]) && cur[1] == '@'))
|
||
{
|
||
if (*cur == '-')
|
||
{
|
||
/* Discard the `-'. */
|
||
++cur;
|
||
got_minus = 1;
|
||
}
|
||
|
||
/* Defining the bitness. */
|
||
switch (*cur)
|
||
{
|
||
case '1':
|
||
b = (got_minus ? STAP_ARG_BITNESS_8BIT_SIGNED
|
||
: STAP_ARG_BITNESS_8BIT_UNSIGNED);
|
||
break;
|
||
|
||
case '2':
|
||
b = (got_minus ? STAP_ARG_BITNESS_16BIT_SIGNED
|
||
: STAP_ARG_BITNESS_16BIT_UNSIGNED);
|
||
break;
|
||
|
||
case '4':
|
||
b = (got_minus ? STAP_ARG_BITNESS_32BIT_SIGNED
|
||
: STAP_ARG_BITNESS_32BIT_UNSIGNED);
|
||
break;
|
||
|
||
case '8':
|
||
b = (got_minus ? STAP_ARG_BITNESS_64BIT_SIGNED
|
||
: STAP_ARG_BITNESS_64BIT_UNSIGNED);
|
||
break;
|
||
|
||
default:
|
||
{
|
||
/* We have an error, because we don't expect anything
|
||
except 1, 2, 4 and 8. */
|
||
warning (_("unrecognized bitness %s%c' for probe `%s'"),
|
||
got_minus ? "`-" : "`", *cur, probe->p.name);
|
||
return;
|
||
}
|
||
}
|
||
|
||
arg.bitness = b;
|
||
|
||
/* Discard the number and the `@' sign. */
|
||
cur += 2;
|
||
}
|
||
else
|
||
arg.bitness = STAP_ARG_BITNESS_UNDEFINED;
|
||
|
||
arg.atype = stap_get_expected_argument_type (gdbarch, arg.bitness,
|
||
probe);
|
||
|
||
expr = stap_parse_argument (&cur, arg.atype, gdbarch);
|
||
|
||
if (stap_expression_debug)
|
||
dump_raw_expression (expr, gdb_stdlog,
|
||
"before conversion to prefix form");
|
||
|
||
prefixify_expression (expr);
|
||
|
||
if (stap_expression_debug)
|
||
dump_prefix_expression (expr, gdb_stdlog);
|
||
|
||
arg.aexpr = expr;
|
||
|
||
/* Start it over again. */
|
||
cur = skip_spaces_const (cur);
|
||
|
||
VEC_safe_push (stap_probe_arg_s, probe->args_u.vec, &arg);
|
||
}
|
||
}
|
||
|
||
/* Implementation of the get_probe_address method. */
|
||
|
||
static CORE_ADDR
|
||
stap_get_probe_address (struct probe *probe, struct objfile *objfile)
|
||
{
|
||
return probe->address + ANOFFSET (objfile->section_offsets,
|
||
SECT_OFF_DATA (objfile));
|
||
}
|
||
|
||
/* Given PROBE, returns the number of arguments present in that probe's
|
||
argument string. */
|
||
|
||
static unsigned
|
||
stap_get_probe_argument_count (struct probe *probe_generic,
|
||
struct frame_info *frame)
|
||
{
|
||
struct stap_probe *probe = (struct stap_probe *) probe_generic;
|
||
struct gdbarch *gdbarch = get_frame_arch (frame);
|
||
|
||
gdb_assert (probe_generic->pops == &stap_probe_ops);
|
||
|
||
if (!probe->args_parsed)
|
||
{
|
||
if (can_evaluate_probe_arguments (probe_generic))
|
||
stap_parse_probe_arguments (probe, gdbarch);
|
||
else
|
||
{
|
||
static int have_warned_stap_incomplete = 0;
|
||
|
||
if (!have_warned_stap_incomplete)
|
||
{
|
||
warning (_(
|
||
"The SystemTap SDT probe support is not fully implemented on this target;\n"
|
||
"you will not be able to inspect the arguments of the probes.\n"
|
||
"Please report a bug against GDB requesting a port to this target."));
|
||
have_warned_stap_incomplete = 1;
|
||
}
|
||
|
||
/* Marking the arguments as "already parsed". */
|
||
probe->args_u.vec = NULL;
|
||
probe->args_parsed = 1;
|
||
}
|
||
}
|
||
|
||
gdb_assert (probe->args_parsed);
|
||
return VEC_length (stap_probe_arg_s, probe->args_u.vec);
|
||
}
|
||
|
||
/* Return 1 if OP is a valid operator inside a probe argument, or zero
|
||
otherwise. */
|
||
|
||
static int
|
||
stap_is_operator (const char *op)
|
||
{
|
||
int ret = 1;
|
||
|
||
switch (*op)
|
||
{
|
||
case '*':
|
||
case '/':
|
||
case '%':
|
||
case '^':
|
||
case '!':
|
||
case '+':
|
||
case '-':
|
||
case '<':
|
||
case '>':
|
||
case '|':
|
||
case '&':
|
||
break;
|
||
|
||
case '=':
|
||
if (op[1] != '=')
|
||
ret = 0;
|
||
break;
|
||
|
||
default:
|
||
/* We didn't find any operator. */
|
||
ret = 0;
|
||
}
|
||
|
||
return ret;
|
||
}
|
||
|
||
/* Return argument N of probe PROBE.
|
||
|
||
If the probe's arguments have not been parsed yet, parse them. If
|
||
there are no arguments, throw an exception (error). Otherwise,
|
||
return the requested argument. */
|
||
|
||
static struct stap_probe_arg *
|
||
stap_get_arg (struct stap_probe *probe, unsigned n, struct gdbarch *gdbarch)
|
||
{
|
||
if (!probe->args_parsed)
|
||
stap_parse_probe_arguments (probe, gdbarch);
|
||
|
||
gdb_assert (probe->args_parsed);
|
||
if (probe->args_u.vec == NULL)
|
||
internal_error (__FILE__, __LINE__,
|
||
_("Probe '%s' apparently does not have arguments, but \n"
|
||
"GDB is requesting its argument number %u anyway. "
|
||
"This should not happen. Please report this bug."),
|
||
probe->p.name, n);
|
||
|
||
return VEC_index (stap_probe_arg_s, probe->args_u.vec, n);
|
||
}
|
||
|
||
/* Implement the `can_evaluate_probe_arguments' method of probe_ops. */
|
||
|
||
static int
|
||
stap_can_evaluate_probe_arguments (struct probe *probe_generic)
|
||
{
|
||
struct stap_probe *stap_probe = (struct stap_probe *) probe_generic;
|
||
struct gdbarch *gdbarch = stap_probe->p.arch;
|
||
|
||
/* For SystemTap probes, we have to guarantee that the method
|
||
stap_is_single_operand is defined on gdbarch. If it is not, then it
|
||
means that argument evaluation is not implemented on this target. */
|
||
return gdbarch_stap_is_single_operand_p (gdbarch);
|
||
}
|
||
|
||
/* Evaluate the probe's argument N (indexed from 0), returning a value
|
||
corresponding to it. Assertion is thrown if N does not exist. */
|
||
|
||
static struct value *
|
||
stap_evaluate_probe_argument (struct probe *probe_generic, unsigned n,
|
||
struct frame_info *frame)
|
||
{
|
||
struct stap_probe *stap_probe = (struct stap_probe *) probe_generic;
|
||
struct gdbarch *gdbarch = get_frame_arch (frame);
|
||
struct stap_probe_arg *arg;
|
||
int pos = 0;
|
||
|
||
gdb_assert (probe_generic->pops == &stap_probe_ops);
|
||
|
||
arg = stap_get_arg (stap_probe, n, gdbarch);
|
||
return evaluate_subexp_standard (arg->atype, arg->aexpr, &pos, EVAL_NORMAL);
|
||
}
|
||
|
||
/* Compile the probe's argument N (indexed from 0) to agent expression.
|
||
Assertion is thrown if N does not exist. */
|
||
|
||
static void
|
||
stap_compile_to_ax (struct probe *probe_generic, struct agent_expr *expr,
|
||
struct axs_value *value, unsigned n)
|
||
{
|
||
struct stap_probe *stap_probe = (struct stap_probe *) probe_generic;
|
||
struct stap_probe_arg *arg;
|
||
union exp_element *pc;
|
||
|
||
gdb_assert (probe_generic->pops == &stap_probe_ops);
|
||
|
||
arg = stap_get_arg (stap_probe, n, expr->gdbarch);
|
||
|
||
pc = arg->aexpr->elts;
|
||
gen_expr (arg->aexpr, &pc, expr, value);
|
||
|
||
require_rvalue (expr, value);
|
||
value->type = arg->atype;
|
||
}
|
||
|
||
/* Destroy (free) the data related to PROBE. PROBE memory itself is not feed
|
||
as it is allocated on an obstack. */
|
||
|
||
static void
|
||
stap_probe_destroy (struct probe *probe_generic)
|
||
{
|
||
struct stap_probe *probe = (struct stap_probe *) probe_generic;
|
||
|
||
gdb_assert (probe_generic->pops == &stap_probe_ops);
|
||
|
||
if (probe->args_parsed)
|
||
{
|
||
struct stap_probe_arg *arg;
|
||
int ix;
|
||
|
||
for (ix = 0; VEC_iterate (stap_probe_arg_s, probe->args_u.vec, ix, arg);
|
||
++ix)
|
||
xfree (arg->aexpr);
|
||
VEC_free (stap_probe_arg_s, probe->args_u.vec);
|
||
}
|
||
}
|
||
|
||
|
||
|
||
/* Set or clear a SystemTap semaphore. ADDRESS is the semaphore's
|
||
address. SET is zero if the semaphore should be cleared, or one
|
||
if it should be set. This is a helper function for `stap_semaphore_down'
|
||
and `stap_semaphore_up'. */
|
||
|
||
static void
|
||
stap_modify_semaphore (CORE_ADDR address, int set, struct gdbarch *gdbarch)
|
||
{
|
||
gdb_byte bytes[sizeof (LONGEST)];
|
||
/* The ABI specifies "unsigned short". */
|
||
struct type *type = builtin_type (gdbarch)->builtin_unsigned_short;
|
||
ULONGEST value;
|
||
|
||
if (address == 0)
|
||
return;
|
||
|
||
/* Swallow errors. */
|
||
if (target_read_memory (address, bytes, TYPE_LENGTH (type)) != 0)
|
||
{
|
||
warning (_("Could not read the value of a SystemTap semaphore."));
|
||
return;
|
||
}
|
||
|
||
value = extract_unsigned_integer (bytes, TYPE_LENGTH (type),
|
||
gdbarch_byte_order (gdbarch));
|
||
/* Note that we explicitly don't worry about overflow or
|
||
underflow. */
|
||
if (set)
|
||
++value;
|
||
else
|
||
--value;
|
||
|
||
store_unsigned_integer (bytes, TYPE_LENGTH (type),
|
||
gdbarch_byte_order (gdbarch), value);
|
||
|
||
if (target_write_memory (address, bytes, TYPE_LENGTH (type)) != 0)
|
||
warning (_("Could not write the value of a SystemTap semaphore."));
|
||
}
|
||
|
||
/* Set a SystemTap semaphore. SEM is the semaphore's address. Semaphores
|
||
act as reference counters, so calls to this function must be paired with
|
||
calls to `stap_semaphore_down'.
|
||
|
||
This function and `stap_semaphore_down' race with another tool changing
|
||
the probes, but that is too rare to care. */
|
||
|
||
static void
|
||
stap_set_semaphore (struct probe *probe_generic, struct objfile *objfile,
|
||
struct gdbarch *gdbarch)
|
||
{
|
||
struct stap_probe *probe = (struct stap_probe *) probe_generic;
|
||
CORE_ADDR addr;
|
||
|
||
gdb_assert (probe_generic->pops == &stap_probe_ops);
|
||
|
||
addr = (probe->sem_addr
|
||
+ ANOFFSET (objfile->section_offsets, SECT_OFF_DATA (objfile)));
|
||
stap_modify_semaphore (addr, 1, gdbarch);
|
||
}
|
||
|
||
/* Clear a SystemTap semaphore. SEM is the semaphore's address. */
|
||
|
||
static void
|
||
stap_clear_semaphore (struct probe *probe_generic, struct objfile *objfile,
|
||
struct gdbarch *gdbarch)
|
||
{
|
||
struct stap_probe *probe = (struct stap_probe *) probe_generic;
|
||
CORE_ADDR addr;
|
||
|
||
gdb_assert (probe_generic->pops == &stap_probe_ops);
|
||
|
||
addr = (probe->sem_addr
|
||
+ ANOFFSET (objfile->section_offsets, SECT_OFF_DATA (objfile)));
|
||
stap_modify_semaphore (addr, 0, gdbarch);
|
||
}
|
||
|
||
/* Helper function that parses the information contained in a
|
||
SystemTap's probe. Basically, the information consists in:
|
||
|
||
- Probe's PC address;
|
||
- Link-time section address of `.stapsdt.base' section;
|
||
- Link-time address of the semaphore variable, or ZERO if the
|
||
probe doesn't have an associated semaphore;
|
||
- Probe's provider name;
|
||
- Probe's name;
|
||
- Probe's argument format
|
||
|
||
This function returns 1 if the handling was successful, and zero
|
||
otherwise. */
|
||
|
||
static void
|
||
handle_stap_probe (struct objfile *objfile, struct sdt_note *el,
|
||
VEC (probe_p) **probesp, CORE_ADDR base)
|
||
{
|
||
bfd *abfd = objfile->obfd;
|
||
int size = bfd_get_arch_size (abfd) / 8;
|
||
struct gdbarch *gdbarch = get_objfile_arch (objfile);
|
||
struct type *ptr_type = builtin_type (gdbarch)->builtin_data_ptr;
|
||
CORE_ADDR base_ref;
|
||
const char *probe_args = NULL;
|
||
struct stap_probe *ret;
|
||
|
||
ret = XOBNEW (&objfile->per_bfd->storage_obstack, struct stap_probe);
|
||
ret->p.pops = &stap_probe_ops;
|
||
ret->p.arch = gdbarch;
|
||
|
||
/* Provider and the name of the probe. */
|
||
ret->p.provider = (char *) &el->data[3 * size];
|
||
ret->p.name = memchr (ret->p.provider, '\0',
|
||
(char *) el->data + el->size - ret->p.provider);
|
||
/* Making sure there is a name. */
|
||
if (ret->p.name == NULL)
|
||
{
|
||
complaint (&symfile_complaints, _("corrupt probe name when "
|
||
"reading `%s'"),
|
||
objfile_name (objfile));
|
||
|
||
/* There is no way to use a probe without a name or a provider, so
|
||
returning zero here makes sense. */
|
||
return;
|
||
}
|
||
else
|
||
++ret->p.name;
|
||
|
||
/* Retrieving the probe's address. */
|
||
ret->p.address = extract_typed_address (&el->data[0], ptr_type);
|
||
|
||
/* Link-time sh_addr of `.stapsdt.base' section. */
|
||
base_ref = extract_typed_address (&el->data[size], ptr_type);
|
||
|
||
/* Semaphore address. */
|
||
ret->sem_addr = extract_typed_address (&el->data[2 * size], ptr_type);
|
||
|
||
ret->p.address += base - base_ref;
|
||
if (ret->sem_addr != 0)
|
||
ret->sem_addr += base - base_ref;
|
||
|
||
/* Arguments. We can only extract the argument format if there is a valid
|
||
name for this probe. */
|
||
probe_args = memchr (ret->p.name, '\0',
|
||
(char *) el->data + el->size - ret->p.name);
|
||
|
||
if (probe_args != NULL)
|
||
++probe_args;
|
||
|
||
if (probe_args == NULL
|
||
|| (memchr (probe_args, '\0', (char *) el->data + el->size - ret->p.name)
|
||
!= el->data + el->size - 1))
|
||
{
|
||
complaint (&symfile_complaints, _("corrupt probe argument when "
|
||
"reading `%s'"),
|
||
objfile_name (objfile));
|
||
/* If the argument string is NULL, it means some problem happened with
|
||
it. So we return 0. */
|
||
return;
|
||
}
|
||
|
||
ret->args_parsed = 0;
|
||
ret->args_u.text = probe_args;
|
||
|
||
/* Successfully created probe. */
|
||
VEC_safe_push (probe_p, *probesp, (struct probe *) ret);
|
||
}
|
||
|
||
/* Helper function which tries to find the base address of the SystemTap
|
||
base section named STAP_BASE_SECTION_NAME. */
|
||
|
||
static void
|
||
get_stap_base_address_1 (bfd *abfd, asection *sect, void *obj)
|
||
{
|
||
asection **ret = (asection **) obj;
|
||
|
||
if ((sect->flags & (SEC_DATA | SEC_ALLOC | SEC_HAS_CONTENTS))
|
||
&& sect->name && !strcmp (sect->name, STAP_BASE_SECTION_NAME))
|
||
*ret = sect;
|
||
}
|
||
|
||
/* Helper function which iterates over every section in the BFD file,
|
||
trying to find the base address of the SystemTap base section.
|
||
Returns 1 if found (setting BASE to the proper value), zero otherwise. */
|
||
|
||
static int
|
||
get_stap_base_address (bfd *obfd, bfd_vma *base)
|
||
{
|
||
asection *ret = NULL;
|
||
|
||
bfd_map_over_sections (obfd, get_stap_base_address_1, (void *) &ret);
|
||
|
||
if (ret == NULL)
|
||
{
|
||
complaint (&symfile_complaints, _("could not obtain base address for "
|
||
"SystemTap section on objfile `%s'."),
|
||
obfd->filename);
|
||
return 0;
|
||
}
|
||
|
||
if (base != NULL)
|
||
*base = ret->vma;
|
||
|
||
return 1;
|
||
}
|
||
|
||
/* Helper function for `elf_get_probes', which gathers information about all
|
||
SystemTap probes from OBJFILE. */
|
||
|
||
static void
|
||
stap_get_probes (VEC (probe_p) **probesp, struct objfile *objfile)
|
||
{
|
||
/* If we are here, then this is the first time we are parsing the
|
||
SystemTap probe's information. We basically have to count how many
|
||
probes the objfile has, and then fill in the necessary information
|
||
for each one. */
|
||
bfd *obfd = objfile->obfd;
|
||
bfd_vma base;
|
||
struct sdt_note *iter;
|
||
unsigned save_probesp_len = VEC_length (probe_p, *probesp);
|
||
|
||
if (objfile->separate_debug_objfile_backlink != NULL)
|
||
{
|
||
/* This is a .debug file, not the objfile itself. */
|
||
return;
|
||
}
|
||
|
||
if (elf_tdata (obfd)->sdt_note_head == NULL)
|
||
{
|
||
/* There isn't any probe here. */
|
||
return;
|
||
}
|
||
|
||
if (!get_stap_base_address (obfd, &base))
|
||
{
|
||
/* There was an error finding the base address for the section.
|
||
Just return NULL. */
|
||
return;
|
||
}
|
||
|
||
/* Parsing each probe's information. */
|
||
for (iter = elf_tdata (obfd)->sdt_note_head;
|
||
iter != NULL;
|
||
iter = iter->next)
|
||
{
|
||
/* We first have to handle all the information about the
|
||
probe which is present in the section. */
|
||
handle_stap_probe (objfile, iter, probesp, base);
|
||
}
|
||
|
||
if (save_probesp_len == VEC_length (probe_p, *probesp))
|
||
{
|
||
/* If we are here, it means we have failed to parse every known
|
||
probe. */
|
||
complaint (&symfile_complaints, _("could not parse SystemTap probe(s) "
|
||
"from inferior"));
|
||
return;
|
||
}
|
||
}
|
||
|
||
/* Implementation of the type_name method. */
|
||
|
||
static const char *
|
||
stap_type_name (struct probe *probe)
|
||
{
|
||
gdb_assert (probe->pops == &stap_probe_ops);
|
||
return "stap";
|
||
}
|
||
|
||
static int
|
||
stap_probe_is_linespec (const char **linespecp)
|
||
{
|
||
static const char *const keywords[] = { "-pstap", "-probe-stap", NULL };
|
||
|
||
return probe_is_linespec_by_keyword (linespecp, keywords);
|
||
}
|
||
|
||
static void
|
||
stap_gen_info_probes_table_header (VEC (info_probe_column_s) **heads)
|
||
{
|
||
info_probe_column_s stap_probe_column;
|
||
|
||
stap_probe_column.field_name = "semaphore";
|
||
stap_probe_column.print_name = _("Semaphore");
|
||
|
||
VEC_safe_push (info_probe_column_s, *heads, &stap_probe_column);
|
||
}
|
||
|
||
static void
|
||
stap_gen_info_probes_table_values (struct probe *probe_generic,
|
||
VEC (const_char_ptr) **ret)
|
||
{
|
||
struct stap_probe *probe = (struct stap_probe *) probe_generic;
|
||
struct gdbarch *gdbarch;
|
||
const char *val = NULL;
|
||
|
||
gdb_assert (probe_generic->pops == &stap_probe_ops);
|
||
|
||
gdbarch = probe->p.arch;
|
||
|
||
if (probe->sem_addr != 0)
|
||
val = print_core_address (gdbarch, probe->sem_addr);
|
||
|
||
VEC_safe_push (const_char_ptr, *ret, val);
|
||
}
|
||
|
||
/* SystemTap probe_ops. */
|
||
|
||
const struct probe_ops stap_probe_ops =
|
||
{
|
||
stap_probe_is_linespec,
|
||
stap_get_probes,
|
||
stap_get_probe_address,
|
||
stap_get_probe_argument_count,
|
||
stap_can_evaluate_probe_arguments,
|
||
stap_evaluate_probe_argument,
|
||
stap_compile_to_ax,
|
||
stap_set_semaphore,
|
||
stap_clear_semaphore,
|
||
stap_probe_destroy,
|
||
stap_type_name,
|
||
stap_gen_info_probes_table_header,
|
||
stap_gen_info_probes_table_values,
|
||
NULL, /* enable_probe */
|
||
NULL /* disable_probe */
|
||
};
|
||
|
||
/* Implementation of the `info probes stap' command. */
|
||
|
||
static void
|
||
info_probes_stap_command (char *arg, int from_tty)
|
||
{
|
||
info_probes_for_ops (arg, from_tty, &stap_probe_ops);
|
||
}
|
||
|
||
void _initialize_stap_probe (void);
|
||
|
||
void
|
||
_initialize_stap_probe (void)
|
||
{
|
||
VEC_safe_push (probe_ops_cp, all_probe_ops, &stap_probe_ops);
|
||
|
||
add_setshow_zuinteger_cmd ("stap-expression", class_maintenance,
|
||
&stap_expression_debug,
|
||
_("Set SystemTap expression debugging."),
|
||
_("Show SystemTap expression debugging."),
|
||
_("When non-zero, the internal representation "
|
||
"of SystemTap expressions will be printed."),
|
||
NULL,
|
||
show_stapexpressiondebug,
|
||
&setdebuglist, &showdebuglist);
|
||
|
||
add_cmd ("stap", class_info, info_probes_stap_command,
|
||
_("\
|
||
Show information about SystemTap static probes.\n\
|
||
Usage: info probes stap [PROVIDER [NAME [OBJECT]]]\n\
|
||
Each argument is a regular expression, used to select probes.\n\
|
||
PROVIDER matches probe provider names.\n\
|
||
NAME matches the probe names.\n\
|
||
OBJECT matches the executable or shared library name."),
|
||
info_probes_cmdlist_get ());
|
||
|
||
}
|