mirror of
https://sourceware.org/git/binutils-gdb.git
synced 2024-12-03 04:12:10 +08:00
9a3c826307
Note: I needed to split this patch in two, otherwise it's too big for the mailing list. This patch adds explicit casts to situations where a void pointer is assigned to a pointer to the "real" type. Building in C++ mode requires those assignments to use an explicit cast. This includes, for example: - callback arguments (cleanups, comparison functions, ...) - data attached to some object (objfile, program space, etc) in the form of a void pointer - "user data" passed to some function This patch comes from the commit "(mostly) auto-generated patch to insert casts needed for C++", taken from Pedro's C++ branch. Only files built on x86 with --enable-targets=all are modified, so the native files for other arches will need to be dealt with separately. I built-tested this with --enable-targets=all and reg-tested. To my surprise, a test case (selftest.exp) had to be adjusted. Here's the ChangeLog entry. Again, this was relatively quick to make despite the length, thanks to David Malcom's script, although I don't believe it's very useful information in that particular case... gdb/ChangeLog: * aarch64-tdep.c (aarch64_make_prologue_cache): Add cast(s). (aarch64_make_stub_cache): Likewise. (value_of_aarch64_user_reg): Likewise. * ada-lang.c (ada_inferior_data_cleanup): Likewise. (get_ada_inferior_data): Likewise. (get_ada_pspace_data): Likewise. (ada_pspace_data_cleanup): Likewise. (ada_complete_symbol_matcher): Likewise. (ada_exc_search_name_matches): Likewise. * ada-tasks.c (get_ada_tasks_pspace_data): Likewise. (get_ada_tasks_inferior_data): Likewise. * addrmap.c (addrmap_mutable_foreach_worker): Likewise. (splay_obstack_alloc): Likewise. (splay_obstack_free): Likewise. * alpha-linux-tdep.c (alpha_linux_supply_gregset): Likewise. (alpha_linux_collect_gregset): Likewise. (alpha_linux_supply_fpregset): Likewise. (alpha_linux_collect_fpregset): Likewise. * alpha-mdebug-tdep.c (alpha_mdebug_frame_unwind_cache): Likewise. * alpha-tdep.c (alpha_lds): Likewise. (alpha_sts): Likewise. (alpha_sigtramp_frame_unwind_cache): Likewise. (alpha_heuristic_frame_unwind_cache): Likewise. (alpha_supply_int_regs): Likewise. (alpha_fill_int_regs): Likewise. (alpha_supply_fp_regs): Likewise. (alpha_fill_fp_regs): Likewise. * alphanbsd-tdep.c (alphanbsd_supply_fpregset): Likewise. (alphanbsd_aout_supply_gregset): Likewise. (alphanbsd_supply_gregset): Likewise. * amd64-linux-tdep.c (amd64_linux_init_abi): Likewise. (amd64_x32_linux_init_abi): Likewise. * amd64-nat.c (amd64_supply_native_gregset): Likewise. (amd64_collect_native_gregset): Likewise. * amd64-tdep.c (amd64_frame_cache): Likewise. (amd64_sigtramp_frame_cache): Likewise. (amd64_epilogue_frame_cache): Likewise. (amd64_supply_fxsave): Likewise. (amd64_supply_xsave): Likewise. (amd64_collect_fxsave): Likewise. (amd64_collect_xsave): Likewise. * amd64-windows-tdep.c (amd64_windows_frame_cache): Likewise. * amd64obsd-tdep.c (amd64obsd_trapframe_cache): Likewise. * arm-linux-tdep.c (arm_linux_supply_gregset): Likewise. (arm_linux_collect_gregset): Likewise. (arm_linux_supply_nwfpe): Likewise. (arm_linux_collect_nwfpe): Likewise. (arm_linux_supply_vfp): Likewise. (arm_linux_collect_vfp): Likewise. * arm-tdep.c (arm_find_mapping_symbol): Likewise. (arm_prologue_unwind_stop_reason): Likewise. (arm_prologue_this_id): Likewise. (arm_prologue_prev_register): Likewise. (arm_exidx_data_free): Likewise. (arm_find_exidx_entry): Likewise. (arm_stub_this_id): Likewise. (arm_m_exception_this_id): Likewise. (arm_m_exception_prev_register): Likewise. (arm_normal_frame_base): Likewise. (gdb_print_insn_arm): Likewise. (arm_objfile_data_free): Likewise. (arm_record_special_symbol): Likewise. (value_of_arm_user_reg): Likewise. * armbsd-tdep.c (armbsd_supply_fpregset): Likewise. (armbsd_supply_gregset): Likewise. * auto-load.c (auto_load_pspace_data_cleanup): Likewise. (get_auto_load_pspace_data): Likewise. (hash_loaded_script_entry): Likewise. (eq_loaded_script_entry): Likewise. (clear_section_scripts): Likewise. (collect_matching_scripts): Likewise. * auxv.c (auxv_inferior_data_cleanup): Likewise. (get_auxv_inferior_data): Likewise. * avr-tdep.c (avr_frame_unwind_cache): Likewise. * ax-general.c (do_free_agent_expr_cleanup): Likewise. * bfd-target.c (target_bfd_xfer_partial): Likewise. (target_bfd_xclose): Likewise. (target_bfd_get_section_table): Likewise. * bfin-tdep.c (bfin_frame_cache): Likewise. * block.c (find_block_in_blockvector): Likewise. (call_site_for_pc): Likewise. (block_find_non_opaque_type_preferred): Likewise. * break-catch-sig.c (signal_catchpoint_insert_location): Likewise. (signal_catchpoint_remove_location): Likewise. (signal_catchpoint_breakpoint_hit): Likewise. (signal_catchpoint_print_one): Likewise. (signal_catchpoint_print_mention): Likewise. (signal_catchpoint_print_recreate): Likewise. * break-catch-syscall.c (get_catch_syscall_inferior_data): Likewise. * breakpoint.c (do_cleanup_counted_command_line): Likewise. (bp_location_compare_addrs): Likewise. (get_first_locp_gte_addr): Likewise. (check_tracepoint_command): Likewise. (do_map_commands_command): Likewise. (get_breakpoint_objfile_data): Likewise. (free_breakpoint_probes): Likewise. (do_captured_breakpoint_query): Likewise. (compare_breakpoints): Likewise. (bp_location_compare): Likewise. (bpstat_remove_breakpoint_callback): Likewise. (do_delete_breakpoint_cleanup): Likewise. * bsd-uthread.c (bsd_uthread_set_supply_uthread): Likewise. (bsd_uthread_set_collect_uthread): Likewise. (bsd_uthread_activate): Likewise. (bsd_uthread_fetch_registers): Likewise. (bsd_uthread_store_registers): Likewise. * btrace.c (check_xml_btrace_version): Likewise. (parse_xml_btrace_block): Likewise. (parse_xml_btrace_pt_config_cpu): Likewise. (parse_xml_btrace_pt_raw): Likewise. (parse_xml_btrace_pt): Likewise. (parse_xml_btrace_conf_bts): Likewise. (parse_xml_btrace_conf_pt): Likewise. (do_btrace_data_cleanup): Likewise. * c-typeprint.c (find_typedef_for_canonicalize): Likewise. * charset.c (cleanup_iconv): Likewise. (do_cleanup_iterator): Likewise. * cli-out.c (cli_uiout_dtor): Likewise. (cli_table_begin): Likewise. (cli_table_body): Likewise. (cli_table_end): Likewise. (cli_table_header): Likewise. (cli_begin): Likewise. (cli_end): Likewise. (cli_field_int): Likewise. (cli_field_skip): Likewise. (cli_field_string): Likewise. (cli_field_fmt): Likewise. (cli_spaces): Likewise. (cli_text): Likewise. (cli_message): Likewise. (cli_wrap_hint): Likewise. (cli_flush): Likewise. (cli_redirect): Likewise. (out_field_fmt): Likewise. (field_separator): Likewise. (cli_out_set_stream): Likewise. * cli/cli-cmds.c (compare_symtabs): Likewise. * cli/cli-dump.c (call_dump_func): Likewise. (restore_section_callback): Likewise. * cli/cli-script.c (clear_hook_in_cleanup): Likewise. (do_restore_user_call_depth): Likewise. (do_free_command_lines_cleanup): Likewise. * coff-pe-read.c (get_section_vmas): Likewise. (pe_as16): Likewise. (pe_as32): Likewise. * coffread.c (coff_symfile_read): Likewise. * common/agent.c (agent_look_up_symbols): Likewise. * common/filestuff.c (do_close_cleanup): Likewise. * common/format.c (free_format_pieces_cleanup): Likewise. * common/vec.c (vec_o_reserve): Likewise. * compile/compile-c-support.c (print_one_macro): Likewise. * compile/compile-c-symbols.c (hash_symbol_error): Likewise. (eq_symbol_error): Likewise. (del_symbol_error): Likewise. (error_symbol_once): Likewise. (gcc_convert_symbol): Likewise. (gcc_symbol_address): Likewise. (hash_symname): Likewise. (eq_symname): Likewise. * compile/compile-c-types.c (hash_type_map_instance): Likewise. (eq_type_map_instance): Likewise. (insert_type): Likewise. (convert_type): Likewise. * compile/compile-object-load.c (munmap_listp_free_cleanup): Likewise. (setup_sections): Likewise. (link_hash_table_free): Likewise. (copy_sections): Likewise. * compile/compile-object-run.c (do_module_cleanup): Likewise. * compile/compile.c (compile_print_value): Likewise. (do_rmdir): Likewise. (cleanup_compile_instance): Likewise. (cleanup_unlink_file): Likewise. * completer.c (free_completion_tracker): Likewise. * corelow.c (add_to_spuid_list): Likewise. * cp-namespace.c (reset_directive_searched): Likewise. * cp-support.c (reset_directive_searched): Likewise. * cris-tdep.c (cris_sigtramp_frame_unwind_cache): Likewise. (cris_frame_unwind_cache): Likewise. * d-lang.c (builtin_d_type): Likewise. * d-namespace.c (reset_directive_searched): Likewise. * dbxread.c (dbx_free_symfile_info): Likewise. (do_free_bincl_list_cleanup): Likewise. * disasm.c (hash_dis_line_entry): Likewise. (eq_dis_line_entry): Likewise. (dis_asm_print_address): Likewise. (fprintf_disasm): Likewise. (do_ui_file_delete): Likewise. * doublest.c (convert_floatformat_to_doublest): Likewise. * dummy-frame.c (pop_dummy_frame_bpt): Likewise. (dummy_frame_prev_register): Likewise. (dummy_frame_this_id): Likewise. * dwarf2-frame-tailcall.c (cache_hash): Likewise. (cache_eq): Likewise. (cache_find): Likewise. (tailcall_frame_this_id): Likewise. (dwarf2_tailcall_prev_register_first): Likewise. (tailcall_frame_prev_register): Likewise. (tailcall_frame_dealloc_cache): Likewise. (tailcall_frame_prev_arch): Likewise. * dwarf2-frame.c (dwarf2_frame_state_free): Likewise. (dwarf2_frame_set_init_reg): Likewise. (dwarf2_frame_init_reg): Likewise. (dwarf2_frame_set_signal_frame_p): Likewise. (dwarf2_frame_signal_frame_p): Likewise. (dwarf2_frame_set_adjust_regnum): Likewise. (dwarf2_frame_adjust_regnum): Likewise. (clear_pointer_cleanup): Likewise. (dwarf2_frame_cache): Likewise. (find_cie): Likewise. (dwarf2_frame_find_fde): Likewise. * dwarf2expr.c (dwarf_expr_address_type): Likewise. (free_dwarf_expr_context_cleanup): Likewise. * dwarf2loc.c (locexpr_find_frame_base_location): Likewise. (locexpr_get_frame_base): Likewise. (loclist_find_frame_base_location): Likewise. (loclist_get_frame_base): Likewise. (dwarf_expr_dwarf_call): Likewise. (dwarf_expr_get_base_type): Likewise. (dwarf_expr_push_dwarf_reg_entry_value): Likewise. (dwarf_expr_get_obj_addr): Likewise. (entry_data_value_coerce_ref): Likewise. (entry_data_value_copy_closure): Likewise. (entry_data_value_free_closure): Likewise. (get_frame_address_in_block_wrapper): Likewise. (dwarf2_evaluate_property): Likewise. (dwarf2_compile_property_to_c): Likewise. (needs_frame_read_addr_from_reg): Likewise. (needs_frame_get_reg_value): Likewise. (needs_frame_frame_base): Likewise. (needs_frame_frame_cfa): Likewise. (needs_frame_tls_address): Likewise. (needs_frame_dwarf_call): Likewise. (needs_dwarf_reg_entry_value): Likewise. (get_ax_pc): Likewise. (locexpr_read_variable): Likewise. (locexpr_read_variable_at_entry): Likewise. (locexpr_read_needs_frame): Likewise. (locexpr_describe_location): Likewise. (locexpr_tracepoint_var_ref): Likewise. (locexpr_generate_c_location): Likewise. (loclist_read_variable): Likewise. (loclist_read_variable_at_entry): Likewise. (loclist_describe_location): Likewise. (loclist_tracepoint_var_ref): Likewise. (loclist_generate_c_location): Likewise. * dwarf2read.c (line_header_hash_voidp): Likewise. (line_header_eq_voidp): Likewise. (dwarf2_has_info): Likewise. (dwarf2_get_section_info): Likewise. (locate_dwz_sections): Likewise. (hash_file_name_entry): Likewise. (eq_file_name_entry): Likewise. (delete_file_name_entry): Likewise. (dw2_setup): Likewise. (dw2_get_file_names_reader): Likewise. (dw2_find_pc_sect_compunit_symtab): Likewise. (hash_signatured_type): Likewise. (eq_signatured_type): Likewise. (add_signatured_type_cu_to_table): Likewise. (create_debug_types_hash_table): Likewise. (lookup_dwo_signatured_type): Likewise. (lookup_dwp_signatured_type): Likewise. (lookup_signatured_type): Likewise. (hash_type_unit_group): Likewise. (eq_type_unit_group): Likewise. (get_type_unit_group): Likewise. (process_psymtab_comp_unit_reader): Likewise. (sort_tu_by_abbrev_offset): Likewise. (process_skeletonless_type_unit): Likewise. (psymtabs_addrmap_cleanup): Likewise. (dwarf2_read_symtab): Likewise. (psymtab_to_symtab_1): Likewise. (die_hash): Likewise. (die_eq): Likewise. (load_full_comp_unit_reader): Likewise. (reset_die_in_process): Likewise. (free_cu_line_header): Likewise. (handle_DW_AT_stmt_list): Likewise. (hash_dwo_file): Likewise. (eq_dwo_file): Likewise. (hash_dwo_unit): Likewise. (eq_dwo_unit): Likewise. (create_dwo_cu_reader): Likewise. (create_dwo_unit_in_dwp_v1): Likewise. (create_dwo_unit_in_dwp_v2): Likewise. (lookup_dwo_unit_in_dwp): Likewise. (dwarf2_locate_dwo_sections): Likewise. (dwarf2_locate_common_dwp_sections): Likewise. (dwarf2_locate_v2_dwp_sections): Likewise. (hash_dwp_loaded_cutus): Likewise. (eq_dwp_loaded_cutus): Likewise. (lookup_dwo_cutu): Likewise. (abbrev_table_free_cleanup): Likewise. (dwarf2_free_abbrev_table): Likewise. (find_partial_die_in_comp_unit): Likewise. (free_line_header_voidp): Likewise. (follow_die_offset): Likewise. (follow_die_sig_1): Likewise. (free_heap_comp_unit): Likewise. (free_stack_comp_unit): Likewise. (dwarf2_free_objfile): Likewise. (per_cu_offset_and_type_hash): Likewise. (per_cu_offset_and_type_eq): Likewise. (get_die_type_at_offset): Likewise. (partial_die_hash): Likewise. (partial_die_eq): Likewise. (dwarf2_per_objfile_free): Likewise. (hash_strtab_entry): Likewise. (eq_strtab_entry): Likewise. (add_string): Likewise. (hash_symtab_entry): Likewise. (eq_symtab_entry): Likewise. (delete_symtab_entry): Likewise. (cleanup_mapped_symtab): Likewise. (add_indices_to_cpool): Likewise. (hash_psymtab_cu_index): Likewise. (eq_psymtab_cu_index): Likewise. (add_address_entry_worker): Likewise. (unlink_if_set): Likewise. (write_one_signatured_type): Likewise. (save_gdb_index_command): Likewise. * elfread.c (elf_symtab_read): Likewise. (elf_gnu_ifunc_cache_hash): Likewise. (elf_gnu_ifunc_cache_eq): Likewise. (elf_gnu_ifunc_record_cache): Likewise. (elf_gnu_ifunc_resolve_by_cache): Likewise. (elf_get_probes): Likewise. (probe_key_free): Likewise. * f-lang.c (builtin_f_type): Likewise. * frame-base.c (frame_base_append_sniffer): Likewise. (frame_base_set_default): Likewise. (frame_base_find_by_frame): Likewise. * frame-unwind.c (frame_unwind_prepend_unwinder): Likewise. (frame_unwind_append_unwinder): Likewise. (frame_unwind_find_by_frame): Likewise. * frame.c (frame_addr_hash): Likewise. (frame_addr_hash_eq): Likewise. (frame_stash_find): Likewise. (do_frame_register_read): Likewise. (unwind_to_current_frame): Likewise. (frame_cleanup_after_sniffer): Likewise. * frv-linux-tdep.c (frv_linux_sigtramp_frame_cache): Likewise. * frv-tdep.c (frv_frame_unwind_cache): Likewise. * ft32-tdep.c (ft32_frame_cache): Likewise. * gcore.c (do_bfd_delete_cleanup): Likewise. (gcore_create_callback): Likewise. * gdb_bfd.c (hash_bfd): Likewise. (eq_bfd): Likewise. (gdb_bfd_open): Likewise. (free_one_bfd_section): Likewise. (gdb_bfd_ref): Likewise. (gdb_bfd_unref): Likewise. (get_section_descriptor): Likewise. (gdb_bfd_map_section): Likewise. (gdb_bfd_crc): Likewise. (gdb_bfd_mark_parent): Likewise. (gdb_bfd_record_inclusion): Likewise. (gdb_bfd_requires_relocations): Likewise. (print_one_bfd): Likewise. * gdbtypes.c (type_pair_hash): Likewise. (type_pair_eq): Likewise. (builtin_type): Likewise. (objfile_type): Likewise. * gnu-v3-abi.c (vtable_ptrdiff_type): Likewise. (vtable_address_point_offset): Likewise. (gnuv3_get_vtable): Likewise. (hash_value_and_voffset): Likewise. (eq_value_and_voffset): Likewise. (compare_value_and_voffset): Likewise. (compute_vtable_size): Likewise. (gnuv3_get_typeid_type): Likewise. * go-lang.c (builtin_go_type): Likewise. * guile/scm-block.c (bkscm_hash_block_smob): Likewise. (bkscm_eq_block_smob): Likewise. (bkscm_objfile_block_map): Likewise. (bkscm_del_objfile_blocks): Likewise. * guile/scm-breakpoint.c (bpscm_build_bp_list): Likewise. * guile/scm-disasm.c (gdbscm_disasm_read_memory_worker): Likewise. (gdbscm_disasm_print_address): Likewise. * guile/scm-frame.c (frscm_hash_frame_smob): Likewise. (frscm_eq_frame_smob): Likewise. (frscm_inferior_frame_map): Likewise. (frscm_del_inferior_frames): Likewise. * guile/scm-gsmob.c (gdbscm_add_objfile_ref): Likewise. * guile/scm-objfile.c (ofscm_handle_objfile_deleted): Likewise. (ofscm_objfile_smob_from_objfile): Likewise. * guile/scm-ports.c (ioscm_write): Likewise. (ioscm_file_port_delete): Likewise. (ioscm_file_port_rewind): Likewise. (ioscm_file_port_put): Likewise. (ioscm_file_port_write): Likewise. * guile/scm-progspace.c (psscm_handle_pspace_deleted): Likewise. (psscm_pspace_smob_from_pspace): Likewise. * guile/scm-safe-call.c (scscm_recording_pre_unwind_handler): Likewise. (scscm_recording_unwind_handler): Likewise. (gdbscm_with_catch): Likewise. (scscm_call_0_body): Likewise. (scscm_call_1_body): Likewise. (scscm_call_2_body): Likewise. (scscm_call_3_body): Likewise. (scscm_call_4_body): Likewise. (scscm_apply_1_body): Likewise. (scscm_eval_scheme_string): Likewise. (gdbscm_safe_eval_string): Likewise. (scscm_source_scheme_script): Likewise. (gdbscm_safe_source_script): Likewise. * guile/scm-string.c (gdbscm_call_scm_to_stringn): Likewise. (gdbscm_call_scm_from_stringn): Likewise. * guile/scm-symbol.c (syscm_hash_symbol_smob): Likewise. (syscm_eq_symbol_smob): Likewise. (syscm_get_symbol_map): Likewise. (syscm_del_objfile_symbols): Likewise. * guile/scm-symtab.c (stscm_hash_symtab_smob): Likewise. (stscm_eq_symtab_smob): Likewise. (stscm_objfile_symtab_map): Likewise. (stscm_del_objfile_symtabs): Likewise. * guile/scm-type.c (tyscm_hash_type_smob): Likewise. (tyscm_eq_type_smob): Likewise. (tyscm_type_map): Likewise. (tyscm_copy_type_recursive): Likewise. (save_objfile_types): Likewise. * guile/scm-utils.c (extract_arg): Likewise. * h8300-tdep.c (h8300_frame_cache): Likewise. * hppa-linux-tdep.c (hppa_linux_sigtramp_frame_unwind_cache): Likewise. * hppa-tdep.c (compare_unwind_entries): Likewise. (find_unwind_entry): Likewise. (hppa_frame_cache): Likewise. (hppa_stub_frame_unwind_cache): Likewise. * hppanbsd-tdep.c (hppanbsd_supply_gregset): Likewise. * hppaobsd-tdep.c (hppaobsd_supply_gregset): Likewise. (hppaobsd_supply_fpregset): Likewise. * i386-cygwin-tdep.c (core_process_module_section): Likewise. * i386-linux-tdep.c (i386_linux_init_abi): Likewise. * i386-tdep.c (i386_frame_cache): Likewise. (i386_epilogue_frame_cache): Likewise. (i386_sigtramp_frame_cache): Likewise. (i386_supply_gregset): Likewise. (i386_collect_gregset): Likewise. (i386_gdbarch_init): Likewise. * i386obsd-tdep.c (i386obsd_aout_supply_regset): Likewise. (i386obsd_trapframe_cache): Likewise. * i387-tdep.c (i387_supply_fsave): Likewise. (i387_collect_fsave): Likewise. (i387_supply_fxsave): Likewise. (i387_collect_fxsave): Likewise. (i387_supply_xsave): Likewise. (i387_collect_xsave): Likewise. * ia64-tdep.c (ia64_frame_cache): Likewise. (ia64_sigtramp_frame_cache): Likewise. * infcmd.c (attach_command_continuation): Likewise. (attach_command_continuation_free_args): Likewise. * inferior.c (restore_inferior): Likewise. (delete_thread_of_inferior): Likewise. * inflow.c (inflow_inferior_data_cleanup): Likewise. (get_inflow_inferior_data): Likewise. (inflow_inferior_exit): Likewise. * infrun.c (displaced_step_clear_cleanup): Likewise. (restore_current_uiout_cleanup): Likewise. (release_stop_context_cleanup): Likewise. (do_restore_infcall_suspend_state_cleanup): Likewise. (do_restore_infcall_control_state_cleanup): Likewise. (restore_inferior_ptid): Likewise. * inline-frame.c (block_starting_point_at): Likewise. * iq2000-tdep.c (iq2000_frame_cache): Likewise. * jit.c (get_jit_objfile_data): Likewise. (get_jit_program_space_data): Likewise. (jit_object_close_impl): Likewise. (jit_find_objf_with_entry_addr): Likewise. (jit_breakpoint_deleted): Likewise. (jit_unwind_reg_set_impl): Likewise. (jit_unwind_reg_get_impl): Likewise. (jit_dealloc_cache): Likewise. (jit_frame_sniffer): Likewise. (jit_frame_prev_register): Likewise. (jit_prepend_unwinder): Likewise. (jit_inferior_exit_hook): Likewise. (free_objfile_data): Likewise. * jv-lang.c (jv_per_objfile_free): Likewise. (get_dynamics_objfile): Likewise. (get_java_class_symtab): Likewise. (builtin_java_type): Likewise. * language.c (language_string_char_type): Likewise. (language_bool_type): Likewise. (language_lookup_primitive_type): Likewise. (language_lookup_primitive_type_as_symbol): Likewise. * linespec.c (hash_address_entry): Likewise. (eq_address_entry): Likewise. (iterate_inline_only): Likewise. (iterate_name_matcher): Likewise. (decode_line_2_compare_items): Likewise. (collect_one_symbol): Likewise. (compare_symbols): Likewise. (compare_msymbols): Likewise. (add_symtabs_to_list): Likewise. (collect_symbols): Likewise. (compare_msyms): Likewise. (add_minsym): Likewise. (cleanup_linespec_result): Likewise. * linux-fork.c (inferior_call_waitpid_cleanup): Likewise. * linux-nat.c (delete_lwp_cleanup): Likewise. (count_events_callback): Likewise. (select_event_lwp_callback): Likewise. (resume_stopped_resumed_lwps): Likewise. * linux-tdep.c (get_linux_gdbarch_data): Likewise. (invalidate_linux_cache_inf): Likewise. (get_linux_inferior_data): Likewise. (linux_find_memory_regions_thunk): Likewise. (linux_make_mappings_callback): Likewise. (linux_corefile_thread_callback): Likewise. (find_mapping_size): Likewise. * linux-thread-db.c (find_new_threads_callback): Likewise. * lm32-tdep.c (lm32_frame_cache): Likewise. * m2-lang.c (builtin_m2_type): Likewise. * m32c-tdep.c (m32c_analyze_frame_prologue): Likewise. * m32r-linux-tdep.c (m32r_linux_sigtramp_frame_cache): Likewise. (m32r_linux_supply_gregset): Likewise. (m32r_linux_collect_gregset): Likewise. * m32r-tdep.c (m32r_frame_unwind_cache): Likewise. * m68hc11-tdep.c (m68hc11_frame_unwind_cache): Likewise. * m68k-tdep.c (m68k_frame_cache): Likewise. * m68kbsd-tdep.c (m68kbsd_supply_fpregset): Likewise. (m68kbsd_supply_gregset): Likewise. * m68klinux-tdep.c (m68k_linux_sigtramp_frame_cache): Likewise. * m88k-tdep.c (m88k_frame_cache): Likewise. (m88k_supply_gregset): Likewise. gdb/gdbserver/ChangeLog: * dll.c (match_dll): Add cast(s). (unloaded_dll): Likewise. * linux-low.c (second_thread_of_pid_p): Likewise. (delete_lwp_callback): Likewise. (count_events_callback): Likewise. (select_event_lwp_callback): Likewise. (linux_set_resume_request): Likewise. * server.c (accumulate_file_name_length): Likewise. (emit_dll_description): Likewise. (handle_qxfer_threads_worker): Likewise. (visit_actioned_threads): Likewise. * thread-db.c (any_thread_of): Likewise. * tracepoint.c (same_process_p): Likewise. (match_blocktype): Likewise. (build_traceframe_info_xml): Likewise. gdb/testsuite/ChangeLog: * gdb.gdb/selftest.exp (do_steps_and_nexts): Adjust expected source line.
5111 lines
147 KiB
C
5111 lines
147 KiB
C
/* Support routines for manipulating internal types for GDB.
|
||
|
||
Copyright (C) 1992-2015 Free Software Foundation, Inc.
|
||
|
||
Contributed by Cygnus Support, using pieces from other GDB modules.
|
||
|
||
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 "bfd.h"
|
||
#include "symtab.h"
|
||
#include "symfile.h"
|
||
#include "objfiles.h"
|
||
#include "gdbtypes.h"
|
||
#include "expression.h"
|
||
#include "language.h"
|
||
#include "target.h"
|
||
#include "value.h"
|
||
#include "demangle.h"
|
||
#include "complaints.h"
|
||
#include "gdbcmd.h"
|
||
#include "cp-abi.h"
|
||
#include "hashtab.h"
|
||
#include "cp-support.h"
|
||
#include "bcache.h"
|
||
#include "dwarf2loc.h"
|
||
#include "gdbcore.h"
|
||
|
||
/* Initialize BADNESS constants. */
|
||
|
||
const struct rank LENGTH_MISMATCH_BADNESS = {100,0};
|
||
|
||
const struct rank TOO_FEW_PARAMS_BADNESS = {100,0};
|
||
const struct rank INCOMPATIBLE_TYPE_BADNESS = {100,0};
|
||
|
||
const struct rank EXACT_MATCH_BADNESS = {0,0};
|
||
|
||
const struct rank INTEGER_PROMOTION_BADNESS = {1,0};
|
||
const struct rank FLOAT_PROMOTION_BADNESS = {1,0};
|
||
const struct rank BASE_PTR_CONVERSION_BADNESS = {1,0};
|
||
const struct rank INTEGER_CONVERSION_BADNESS = {2,0};
|
||
const struct rank FLOAT_CONVERSION_BADNESS = {2,0};
|
||
const struct rank INT_FLOAT_CONVERSION_BADNESS = {2,0};
|
||
const struct rank VOID_PTR_CONVERSION_BADNESS = {2,0};
|
||
const struct rank BOOL_CONVERSION_BADNESS = {3,0};
|
||
const struct rank BASE_CONVERSION_BADNESS = {2,0};
|
||
const struct rank REFERENCE_CONVERSION_BADNESS = {2,0};
|
||
const struct rank NULL_POINTER_CONVERSION_BADNESS = {2,0};
|
||
const struct rank NS_POINTER_CONVERSION_BADNESS = {10,0};
|
||
const struct rank NS_INTEGER_POINTER_CONVERSION_BADNESS = {3,0};
|
||
|
||
/* Floatformat pairs. */
|
||
const struct floatformat *floatformats_ieee_half[BFD_ENDIAN_UNKNOWN] = {
|
||
&floatformat_ieee_half_big,
|
||
&floatformat_ieee_half_little
|
||
};
|
||
const struct floatformat *floatformats_ieee_single[BFD_ENDIAN_UNKNOWN] = {
|
||
&floatformat_ieee_single_big,
|
||
&floatformat_ieee_single_little
|
||
};
|
||
const struct floatformat *floatformats_ieee_double[BFD_ENDIAN_UNKNOWN] = {
|
||
&floatformat_ieee_double_big,
|
||
&floatformat_ieee_double_little
|
||
};
|
||
const struct floatformat *floatformats_ieee_double_littlebyte_bigword[BFD_ENDIAN_UNKNOWN] = {
|
||
&floatformat_ieee_double_big,
|
||
&floatformat_ieee_double_littlebyte_bigword
|
||
};
|
||
const struct floatformat *floatformats_i387_ext[BFD_ENDIAN_UNKNOWN] = {
|
||
&floatformat_i387_ext,
|
||
&floatformat_i387_ext
|
||
};
|
||
const struct floatformat *floatformats_m68881_ext[BFD_ENDIAN_UNKNOWN] = {
|
||
&floatformat_m68881_ext,
|
||
&floatformat_m68881_ext
|
||
};
|
||
const struct floatformat *floatformats_arm_ext[BFD_ENDIAN_UNKNOWN] = {
|
||
&floatformat_arm_ext_big,
|
||
&floatformat_arm_ext_littlebyte_bigword
|
||
};
|
||
const struct floatformat *floatformats_ia64_spill[BFD_ENDIAN_UNKNOWN] = {
|
||
&floatformat_ia64_spill_big,
|
||
&floatformat_ia64_spill_little
|
||
};
|
||
const struct floatformat *floatformats_ia64_quad[BFD_ENDIAN_UNKNOWN] = {
|
||
&floatformat_ia64_quad_big,
|
||
&floatformat_ia64_quad_little
|
||
};
|
||
const struct floatformat *floatformats_vax_f[BFD_ENDIAN_UNKNOWN] = {
|
||
&floatformat_vax_f,
|
||
&floatformat_vax_f
|
||
};
|
||
const struct floatformat *floatformats_vax_d[BFD_ENDIAN_UNKNOWN] = {
|
||
&floatformat_vax_d,
|
||
&floatformat_vax_d
|
||
};
|
||
const struct floatformat *floatformats_ibm_long_double[BFD_ENDIAN_UNKNOWN] = {
|
||
&floatformat_ibm_long_double_big,
|
||
&floatformat_ibm_long_double_little
|
||
};
|
||
|
||
/* Should opaque types be resolved? */
|
||
|
||
static int opaque_type_resolution = 1;
|
||
|
||
/* A flag to enable printing of debugging information of C++
|
||
overloading. */
|
||
|
||
unsigned int overload_debug = 0;
|
||
|
||
/* A flag to enable strict type checking. */
|
||
|
||
static int strict_type_checking = 1;
|
||
|
||
/* A function to show whether opaque types are resolved. */
|
||
|
||
static void
|
||
show_opaque_type_resolution (struct ui_file *file, int from_tty,
|
||
struct cmd_list_element *c,
|
||
const char *value)
|
||
{
|
||
fprintf_filtered (file, _("Resolution of opaque struct/class/union types "
|
||
"(if set before loading symbols) is %s.\n"),
|
||
value);
|
||
}
|
||
|
||
/* A function to show whether C++ overload debugging is enabled. */
|
||
|
||
static void
|
||
show_overload_debug (struct ui_file *file, int from_tty,
|
||
struct cmd_list_element *c, const char *value)
|
||
{
|
||
fprintf_filtered (file, _("Debugging of C++ overloading is %s.\n"),
|
||
value);
|
||
}
|
||
|
||
/* A function to show the status of strict type checking. */
|
||
|
||
static void
|
||
show_strict_type_checking (struct ui_file *file, int from_tty,
|
||
struct cmd_list_element *c, const char *value)
|
||
{
|
||
fprintf_filtered (file, _("Strict type checking is %s.\n"), value);
|
||
}
|
||
|
||
|
||
/* Allocate a new OBJFILE-associated type structure and fill it
|
||
with some defaults. Space for the type structure is allocated
|
||
on the objfile's objfile_obstack. */
|
||
|
||
struct type *
|
||
alloc_type (struct objfile *objfile)
|
||
{
|
||
struct type *type;
|
||
|
||
gdb_assert (objfile != NULL);
|
||
|
||
/* Alloc the structure and start off with all fields zeroed. */
|
||
type = OBSTACK_ZALLOC (&objfile->objfile_obstack, struct type);
|
||
TYPE_MAIN_TYPE (type) = OBSTACK_ZALLOC (&objfile->objfile_obstack,
|
||
struct main_type);
|
||
OBJSTAT (objfile, n_types++);
|
||
|
||
TYPE_OBJFILE_OWNED (type) = 1;
|
||
TYPE_OWNER (type).objfile = objfile;
|
||
|
||
/* Initialize the fields that might not be zero. */
|
||
|
||
TYPE_CODE (type) = TYPE_CODE_UNDEF;
|
||
TYPE_CHAIN (type) = type; /* Chain back to itself. */
|
||
|
||
return type;
|
||
}
|
||
|
||
/* Allocate a new GDBARCH-associated type structure and fill it
|
||
with some defaults. Space for the type structure is allocated
|
||
on the obstack associated with GDBARCH. */
|
||
|
||
struct type *
|
||
alloc_type_arch (struct gdbarch *gdbarch)
|
||
{
|
||
struct type *type;
|
||
|
||
gdb_assert (gdbarch != NULL);
|
||
|
||
/* Alloc the structure and start off with all fields zeroed. */
|
||
|
||
type = GDBARCH_OBSTACK_ZALLOC (gdbarch, struct type);
|
||
TYPE_MAIN_TYPE (type) = GDBARCH_OBSTACK_ZALLOC (gdbarch, struct main_type);
|
||
|
||
TYPE_OBJFILE_OWNED (type) = 0;
|
||
TYPE_OWNER (type).gdbarch = gdbarch;
|
||
|
||
/* Initialize the fields that might not be zero. */
|
||
|
||
TYPE_CODE (type) = TYPE_CODE_UNDEF;
|
||
TYPE_CHAIN (type) = type; /* Chain back to itself. */
|
||
|
||
return type;
|
||
}
|
||
|
||
/* If TYPE is objfile-associated, allocate a new type structure
|
||
associated with the same objfile. If TYPE is gdbarch-associated,
|
||
allocate a new type structure associated with the same gdbarch. */
|
||
|
||
struct type *
|
||
alloc_type_copy (const struct type *type)
|
||
{
|
||
if (TYPE_OBJFILE_OWNED (type))
|
||
return alloc_type (TYPE_OWNER (type).objfile);
|
||
else
|
||
return alloc_type_arch (TYPE_OWNER (type).gdbarch);
|
||
}
|
||
|
||
/* If TYPE is gdbarch-associated, return that architecture.
|
||
If TYPE is objfile-associated, return that objfile's architecture. */
|
||
|
||
struct gdbarch *
|
||
get_type_arch (const struct type *type)
|
||
{
|
||
if (TYPE_OBJFILE_OWNED (type))
|
||
return get_objfile_arch (TYPE_OWNER (type).objfile);
|
||
else
|
||
return TYPE_OWNER (type).gdbarch;
|
||
}
|
||
|
||
/* See gdbtypes.h. */
|
||
|
||
struct type *
|
||
get_target_type (struct type *type)
|
||
{
|
||
if (type != NULL)
|
||
{
|
||
type = TYPE_TARGET_TYPE (type);
|
||
if (type != NULL)
|
||
type = check_typedef (type);
|
||
}
|
||
|
||
return type;
|
||
}
|
||
|
||
/* See gdbtypes.h. */
|
||
|
||
unsigned int
|
||
type_length_units (struct type *type)
|
||
{
|
||
struct gdbarch *arch = get_type_arch (type);
|
||
int unit_size = gdbarch_addressable_memory_unit_size (arch);
|
||
|
||
return TYPE_LENGTH (type) / unit_size;
|
||
}
|
||
|
||
/* Alloc a new type instance structure, fill it with some defaults,
|
||
and point it at OLDTYPE. Allocate the new type instance from the
|
||
same place as OLDTYPE. */
|
||
|
||
static struct type *
|
||
alloc_type_instance (struct type *oldtype)
|
||
{
|
||
struct type *type;
|
||
|
||
/* Allocate the structure. */
|
||
|
||
if (! TYPE_OBJFILE_OWNED (oldtype))
|
||
type = XCNEW (struct type);
|
||
else
|
||
type = OBSTACK_ZALLOC (&TYPE_OBJFILE (oldtype)->objfile_obstack,
|
||
struct type);
|
||
|
||
TYPE_MAIN_TYPE (type) = TYPE_MAIN_TYPE (oldtype);
|
||
|
||
TYPE_CHAIN (type) = type; /* Chain back to itself for now. */
|
||
|
||
return type;
|
||
}
|
||
|
||
/* Clear all remnants of the previous type at TYPE, in preparation for
|
||
replacing it with something else. Preserve owner information. */
|
||
|
||
static void
|
||
smash_type (struct type *type)
|
||
{
|
||
int objfile_owned = TYPE_OBJFILE_OWNED (type);
|
||
union type_owner owner = TYPE_OWNER (type);
|
||
|
||
memset (TYPE_MAIN_TYPE (type), 0, sizeof (struct main_type));
|
||
|
||
/* Restore owner information. */
|
||
TYPE_OBJFILE_OWNED (type) = objfile_owned;
|
||
TYPE_OWNER (type) = owner;
|
||
|
||
/* For now, delete the rings. */
|
||
TYPE_CHAIN (type) = type;
|
||
|
||
/* For now, leave the pointer/reference types alone. */
|
||
}
|
||
|
||
/* Lookup a pointer to a type TYPE. TYPEPTR, if nonzero, points
|
||
to a pointer to memory where the pointer type should be stored.
|
||
If *TYPEPTR is zero, update it to point to the pointer type we return.
|
||
We allocate new memory if needed. */
|
||
|
||
struct type *
|
||
make_pointer_type (struct type *type, struct type **typeptr)
|
||
{
|
||
struct type *ntype; /* New type */
|
||
struct type *chain;
|
||
|
||
ntype = TYPE_POINTER_TYPE (type);
|
||
|
||
if (ntype)
|
||
{
|
||
if (typeptr == 0)
|
||
return ntype; /* Don't care about alloc,
|
||
and have new type. */
|
||
else if (*typeptr == 0)
|
||
{
|
||
*typeptr = ntype; /* Tracking alloc, and have new type. */
|
||
return ntype;
|
||
}
|
||
}
|
||
|
||
if (typeptr == 0 || *typeptr == 0) /* We'll need to allocate one. */
|
||
{
|
||
ntype = alloc_type_copy (type);
|
||
if (typeptr)
|
||
*typeptr = ntype;
|
||
}
|
||
else /* We have storage, but need to reset it. */
|
||
{
|
||
ntype = *typeptr;
|
||
chain = TYPE_CHAIN (ntype);
|
||
smash_type (ntype);
|
||
TYPE_CHAIN (ntype) = chain;
|
||
}
|
||
|
||
TYPE_TARGET_TYPE (ntype) = type;
|
||
TYPE_POINTER_TYPE (type) = ntype;
|
||
|
||
/* FIXME! Assumes the machine has only one representation for pointers! */
|
||
|
||
TYPE_LENGTH (ntype)
|
||
= gdbarch_ptr_bit (get_type_arch (type)) / TARGET_CHAR_BIT;
|
||
TYPE_CODE (ntype) = TYPE_CODE_PTR;
|
||
|
||
/* Mark pointers as unsigned. The target converts between pointers
|
||
and addresses (CORE_ADDRs) using gdbarch_pointer_to_address and
|
||
gdbarch_address_to_pointer. */
|
||
TYPE_UNSIGNED (ntype) = 1;
|
||
|
||
/* Update the length of all the other variants of this type. */
|
||
chain = TYPE_CHAIN (ntype);
|
||
while (chain != ntype)
|
||
{
|
||
TYPE_LENGTH (chain) = TYPE_LENGTH (ntype);
|
||
chain = TYPE_CHAIN (chain);
|
||
}
|
||
|
||
return ntype;
|
||
}
|
||
|
||
/* Given a type TYPE, return a type of pointers to that type.
|
||
May need to construct such a type if this is the first use. */
|
||
|
||
struct type *
|
||
lookup_pointer_type (struct type *type)
|
||
{
|
||
return make_pointer_type (type, (struct type **) 0);
|
||
}
|
||
|
||
/* Lookup a C++ `reference' to a type TYPE. TYPEPTR, if nonzero,
|
||
points to a pointer to memory where the reference type should be
|
||
stored. If *TYPEPTR is zero, update it to point to the reference
|
||
type we return. We allocate new memory if needed. */
|
||
|
||
struct type *
|
||
make_reference_type (struct type *type, struct type **typeptr)
|
||
{
|
||
struct type *ntype; /* New type */
|
||
struct type *chain;
|
||
|
||
ntype = TYPE_REFERENCE_TYPE (type);
|
||
|
||
if (ntype)
|
||
{
|
||
if (typeptr == 0)
|
||
return ntype; /* Don't care about alloc,
|
||
and have new type. */
|
||
else if (*typeptr == 0)
|
||
{
|
||
*typeptr = ntype; /* Tracking alloc, and have new type. */
|
||
return ntype;
|
||
}
|
||
}
|
||
|
||
if (typeptr == 0 || *typeptr == 0) /* We'll need to allocate one. */
|
||
{
|
||
ntype = alloc_type_copy (type);
|
||
if (typeptr)
|
||
*typeptr = ntype;
|
||
}
|
||
else /* We have storage, but need to reset it. */
|
||
{
|
||
ntype = *typeptr;
|
||
chain = TYPE_CHAIN (ntype);
|
||
smash_type (ntype);
|
||
TYPE_CHAIN (ntype) = chain;
|
||
}
|
||
|
||
TYPE_TARGET_TYPE (ntype) = type;
|
||
TYPE_REFERENCE_TYPE (type) = ntype;
|
||
|
||
/* FIXME! Assume the machine has only one representation for
|
||
references, and that it matches the (only) representation for
|
||
pointers! */
|
||
|
||
TYPE_LENGTH (ntype) =
|
||
gdbarch_ptr_bit (get_type_arch (type)) / TARGET_CHAR_BIT;
|
||
TYPE_CODE (ntype) = TYPE_CODE_REF;
|
||
|
||
if (!TYPE_REFERENCE_TYPE (type)) /* Remember it, if don't have one. */
|
||
TYPE_REFERENCE_TYPE (type) = ntype;
|
||
|
||
/* Update the length of all the other variants of this type. */
|
||
chain = TYPE_CHAIN (ntype);
|
||
while (chain != ntype)
|
||
{
|
||
TYPE_LENGTH (chain) = TYPE_LENGTH (ntype);
|
||
chain = TYPE_CHAIN (chain);
|
||
}
|
||
|
||
return ntype;
|
||
}
|
||
|
||
/* Same as above, but caller doesn't care about memory allocation
|
||
details. */
|
||
|
||
struct type *
|
||
lookup_reference_type (struct type *type)
|
||
{
|
||
return make_reference_type (type, (struct type **) 0);
|
||
}
|
||
|
||
/* Lookup a function type that returns type TYPE. TYPEPTR, if
|
||
nonzero, points to a pointer to memory where the function type
|
||
should be stored. If *TYPEPTR is zero, update it to point to the
|
||
function type we return. We allocate new memory if needed. */
|
||
|
||
struct type *
|
||
make_function_type (struct type *type, struct type **typeptr)
|
||
{
|
||
struct type *ntype; /* New type */
|
||
|
||
if (typeptr == 0 || *typeptr == 0) /* We'll need to allocate one. */
|
||
{
|
||
ntype = alloc_type_copy (type);
|
||
if (typeptr)
|
||
*typeptr = ntype;
|
||
}
|
||
else /* We have storage, but need to reset it. */
|
||
{
|
||
ntype = *typeptr;
|
||
smash_type (ntype);
|
||
}
|
||
|
||
TYPE_TARGET_TYPE (ntype) = type;
|
||
|
||
TYPE_LENGTH (ntype) = 1;
|
||
TYPE_CODE (ntype) = TYPE_CODE_FUNC;
|
||
|
||
INIT_FUNC_SPECIFIC (ntype);
|
||
|
||
return ntype;
|
||
}
|
||
|
||
/* Given a type TYPE, return a type of functions that return that type.
|
||
May need to construct such a type if this is the first use. */
|
||
|
||
struct type *
|
||
lookup_function_type (struct type *type)
|
||
{
|
||
return make_function_type (type, (struct type **) 0);
|
||
}
|
||
|
||
/* Given a type TYPE and argument types, return the appropriate
|
||
function type. If the final type in PARAM_TYPES is NULL, make a
|
||
varargs function. */
|
||
|
||
struct type *
|
||
lookup_function_type_with_arguments (struct type *type,
|
||
int nparams,
|
||
struct type **param_types)
|
||
{
|
||
struct type *fn = make_function_type (type, (struct type **) 0);
|
||
int i;
|
||
|
||
if (nparams > 0)
|
||
{
|
||
if (param_types[nparams - 1] == NULL)
|
||
{
|
||
--nparams;
|
||
TYPE_VARARGS (fn) = 1;
|
||
}
|
||
else if (TYPE_CODE (check_typedef (param_types[nparams - 1]))
|
||
== TYPE_CODE_VOID)
|
||
{
|
||
--nparams;
|
||
/* Caller should have ensured this. */
|
||
gdb_assert (nparams == 0);
|
||
TYPE_PROTOTYPED (fn) = 1;
|
||
}
|
||
}
|
||
|
||
TYPE_NFIELDS (fn) = nparams;
|
||
TYPE_FIELDS (fn)
|
||
= (struct field *) TYPE_ZALLOC (fn, nparams * sizeof (struct field));
|
||
for (i = 0; i < nparams; ++i)
|
||
TYPE_FIELD_TYPE (fn, i) = param_types[i];
|
||
|
||
return fn;
|
||
}
|
||
|
||
/* Identify address space identifier by name --
|
||
return the integer flag defined in gdbtypes.h. */
|
||
|
||
int
|
||
address_space_name_to_int (struct gdbarch *gdbarch, char *space_identifier)
|
||
{
|
||
int type_flags;
|
||
|
||
/* Check for known address space delimiters. */
|
||
if (!strcmp (space_identifier, "code"))
|
||
return TYPE_INSTANCE_FLAG_CODE_SPACE;
|
||
else if (!strcmp (space_identifier, "data"))
|
||
return TYPE_INSTANCE_FLAG_DATA_SPACE;
|
||
else if (gdbarch_address_class_name_to_type_flags_p (gdbarch)
|
||
&& gdbarch_address_class_name_to_type_flags (gdbarch,
|
||
space_identifier,
|
||
&type_flags))
|
||
return type_flags;
|
||
else
|
||
error (_("Unknown address space specifier: \"%s\""), space_identifier);
|
||
}
|
||
|
||
/* Identify address space identifier by integer flag as defined in
|
||
gdbtypes.h -- return the string version of the adress space name. */
|
||
|
||
const char *
|
||
address_space_int_to_name (struct gdbarch *gdbarch, int space_flag)
|
||
{
|
||
if (space_flag & TYPE_INSTANCE_FLAG_CODE_SPACE)
|
||
return "code";
|
||
else if (space_flag & TYPE_INSTANCE_FLAG_DATA_SPACE)
|
||
return "data";
|
||
else if ((space_flag & TYPE_INSTANCE_FLAG_ADDRESS_CLASS_ALL)
|
||
&& gdbarch_address_class_type_flags_to_name_p (gdbarch))
|
||
return gdbarch_address_class_type_flags_to_name (gdbarch, space_flag);
|
||
else
|
||
return NULL;
|
||
}
|
||
|
||
/* Create a new type with instance flags NEW_FLAGS, based on TYPE.
|
||
|
||
If STORAGE is non-NULL, create the new type instance there.
|
||
STORAGE must be in the same obstack as TYPE. */
|
||
|
||
static struct type *
|
||
make_qualified_type (struct type *type, int new_flags,
|
||
struct type *storage)
|
||
{
|
||
struct type *ntype;
|
||
|
||
ntype = type;
|
||
do
|
||
{
|
||
if (TYPE_INSTANCE_FLAGS (ntype) == new_flags)
|
||
return ntype;
|
||
ntype = TYPE_CHAIN (ntype);
|
||
}
|
||
while (ntype != type);
|
||
|
||
/* Create a new type instance. */
|
||
if (storage == NULL)
|
||
ntype = alloc_type_instance (type);
|
||
else
|
||
{
|
||
/* If STORAGE was provided, it had better be in the same objfile
|
||
as TYPE. Otherwise, we can't link it into TYPE's cv chain:
|
||
if one objfile is freed and the other kept, we'd have
|
||
dangling pointers. */
|
||
gdb_assert (TYPE_OBJFILE (type) == TYPE_OBJFILE (storage));
|
||
|
||
ntype = storage;
|
||
TYPE_MAIN_TYPE (ntype) = TYPE_MAIN_TYPE (type);
|
||
TYPE_CHAIN (ntype) = ntype;
|
||
}
|
||
|
||
/* Pointers or references to the original type are not relevant to
|
||
the new type. */
|
||
TYPE_POINTER_TYPE (ntype) = (struct type *) 0;
|
||
TYPE_REFERENCE_TYPE (ntype) = (struct type *) 0;
|
||
|
||
/* Chain the new qualified type to the old type. */
|
||
TYPE_CHAIN (ntype) = TYPE_CHAIN (type);
|
||
TYPE_CHAIN (type) = ntype;
|
||
|
||
/* Now set the instance flags and return the new type. */
|
||
TYPE_INSTANCE_FLAGS (ntype) = new_flags;
|
||
|
||
/* Set length of new type to that of the original type. */
|
||
TYPE_LENGTH (ntype) = TYPE_LENGTH (type);
|
||
|
||
return ntype;
|
||
}
|
||
|
||
/* Make an address-space-delimited variant of a type -- a type that
|
||
is identical to the one supplied except that it has an address
|
||
space attribute attached to it (such as "code" or "data").
|
||
|
||
The space attributes "code" and "data" are for Harvard
|
||
architectures. The address space attributes are for architectures
|
||
which have alternately sized pointers or pointers with alternate
|
||
representations. */
|
||
|
||
struct type *
|
||
make_type_with_address_space (struct type *type, int space_flag)
|
||
{
|
||
int new_flags = ((TYPE_INSTANCE_FLAGS (type)
|
||
& ~(TYPE_INSTANCE_FLAG_CODE_SPACE
|
||
| TYPE_INSTANCE_FLAG_DATA_SPACE
|
||
| TYPE_INSTANCE_FLAG_ADDRESS_CLASS_ALL))
|
||
| space_flag);
|
||
|
||
return make_qualified_type (type, new_flags, NULL);
|
||
}
|
||
|
||
/* Make a "c-v" variant of a type -- a type that is identical to the
|
||
one supplied except that it may have const or volatile attributes
|
||
CNST is a flag for setting the const attribute
|
||
VOLTL is a flag for setting the volatile attribute
|
||
TYPE is the base type whose variant we are creating.
|
||
|
||
If TYPEPTR and *TYPEPTR are non-zero, then *TYPEPTR points to
|
||
storage to hold the new qualified type; *TYPEPTR and TYPE must be
|
||
in the same objfile. Otherwise, allocate fresh memory for the new
|
||
type whereever TYPE lives. If TYPEPTR is non-zero, set it to the
|
||
new type we construct. */
|
||
|
||
struct type *
|
||
make_cv_type (int cnst, int voltl,
|
||
struct type *type,
|
||
struct type **typeptr)
|
||
{
|
||
struct type *ntype; /* New type */
|
||
|
||
int new_flags = (TYPE_INSTANCE_FLAGS (type)
|
||
& ~(TYPE_INSTANCE_FLAG_CONST
|
||
| TYPE_INSTANCE_FLAG_VOLATILE));
|
||
|
||
if (cnst)
|
||
new_flags |= TYPE_INSTANCE_FLAG_CONST;
|
||
|
||
if (voltl)
|
||
new_flags |= TYPE_INSTANCE_FLAG_VOLATILE;
|
||
|
||
if (typeptr && *typeptr != NULL)
|
||
{
|
||
/* TYPE and *TYPEPTR must be in the same objfile. We can't have
|
||
a C-V variant chain that threads across objfiles: if one
|
||
objfile gets freed, then the other has a broken C-V chain.
|
||
|
||
This code used to try to copy over the main type from TYPE to
|
||
*TYPEPTR if they were in different objfiles, but that's
|
||
wrong, too: TYPE may have a field list or member function
|
||
lists, which refer to types of their own, etc. etc. The
|
||
whole shebang would need to be copied over recursively; you
|
||
can't have inter-objfile pointers. The only thing to do is
|
||
to leave stub types as stub types, and look them up afresh by
|
||
name each time you encounter them. */
|
||
gdb_assert (TYPE_OBJFILE (*typeptr) == TYPE_OBJFILE (type));
|
||
}
|
||
|
||
ntype = make_qualified_type (type, new_flags,
|
||
typeptr ? *typeptr : NULL);
|
||
|
||
if (typeptr != NULL)
|
||
*typeptr = ntype;
|
||
|
||
return ntype;
|
||
}
|
||
|
||
/* Make a 'restrict'-qualified version of TYPE. */
|
||
|
||
struct type *
|
||
make_restrict_type (struct type *type)
|
||
{
|
||
return make_qualified_type (type,
|
||
(TYPE_INSTANCE_FLAGS (type)
|
||
| TYPE_INSTANCE_FLAG_RESTRICT),
|
||
NULL);
|
||
}
|
||
|
||
/* Make a type without const, volatile, or restrict. */
|
||
|
||
struct type *
|
||
make_unqualified_type (struct type *type)
|
||
{
|
||
return make_qualified_type (type,
|
||
(TYPE_INSTANCE_FLAGS (type)
|
||
& ~(TYPE_INSTANCE_FLAG_CONST
|
||
| TYPE_INSTANCE_FLAG_VOLATILE
|
||
| TYPE_INSTANCE_FLAG_RESTRICT)),
|
||
NULL);
|
||
}
|
||
|
||
/* Make a '_Atomic'-qualified version of TYPE. */
|
||
|
||
struct type *
|
||
make_atomic_type (struct type *type)
|
||
{
|
||
return make_qualified_type (type,
|
||
(TYPE_INSTANCE_FLAGS (type)
|
||
| TYPE_INSTANCE_FLAG_ATOMIC),
|
||
NULL);
|
||
}
|
||
|
||
/* Replace the contents of ntype with the type *type. This changes the
|
||
contents, rather than the pointer for TYPE_MAIN_TYPE (ntype); thus
|
||
the changes are propogated to all types in the TYPE_CHAIN.
|
||
|
||
In order to build recursive types, it's inevitable that we'll need
|
||
to update types in place --- but this sort of indiscriminate
|
||
smashing is ugly, and needs to be replaced with something more
|
||
controlled. TYPE_MAIN_TYPE is a step in this direction; it's not
|
||
clear if more steps are needed. */
|
||
|
||
void
|
||
replace_type (struct type *ntype, struct type *type)
|
||
{
|
||
struct type *chain;
|
||
|
||
/* These two types had better be in the same objfile. Otherwise,
|
||
the assignment of one type's main type structure to the other
|
||
will produce a type with references to objects (names; field
|
||
lists; etc.) allocated on an objfile other than its own. */
|
||
gdb_assert (TYPE_OBJFILE (ntype) == TYPE_OBJFILE (ntype));
|
||
|
||
*TYPE_MAIN_TYPE (ntype) = *TYPE_MAIN_TYPE (type);
|
||
|
||
/* The type length is not a part of the main type. Update it for
|
||
each type on the variant chain. */
|
||
chain = ntype;
|
||
do
|
||
{
|
||
/* Assert that this element of the chain has no address-class bits
|
||
set in its flags. Such type variants might have type lengths
|
||
which are supposed to be different from the non-address-class
|
||
variants. This assertion shouldn't ever be triggered because
|
||
symbol readers which do construct address-class variants don't
|
||
call replace_type(). */
|
||
gdb_assert (TYPE_ADDRESS_CLASS_ALL (chain) == 0);
|
||
|
||
TYPE_LENGTH (chain) = TYPE_LENGTH (type);
|
||
chain = TYPE_CHAIN (chain);
|
||
}
|
||
while (ntype != chain);
|
||
|
||
/* Assert that the two types have equivalent instance qualifiers.
|
||
This should be true for at least all of our debug readers. */
|
||
gdb_assert (TYPE_INSTANCE_FLAGS (ntype) == TYPE_INSTANCE_FLAGS (type));
|
||
}
|
||
|
||
/* Implement direct support for MEMBER_TYPE in GNU C++.
|
||
May need to construct such a type if this is the first use.
|
||
The TYPE is the type of the member. The DOMAIN is the type
|
||
of the aggregate that the member belongs to. */
|
||
|
||
struct type *
|
||
lookup_memberptr_type (struct type *type, struct type *domain)
|
||
{
|
||
struct type *mtype;
|
||
|
||
mtype = alloc_type_copy (type);
|
||
smash_to_memberptr_type (mtype, domain, type);
|
||
return mtype;
|
||
}
|
||
|
||
/* Return a pointer-to-method type, for a method of type TO_TYPE. */
|
||
|
||
struct type *
|
||
lookup_methodptr_type (struct type *to_type)
|
||
{
|
||
struct type *mtype;
|
||
|
||
mtype = alloc_type_copy (to_type);
|
||
smash_to_methodptr_type (mtype, to_type);
|
||
return mtype;
|
||
}
|
||
|
||
/* Allocate a stub method whose return type is TYPE. This apparently
|
||
happens for speed of symbol reading, since parsing out the
|
||
arguments to the method is cpu-intensive, the way we are doing it.
|
||
So, we will fill in arguments later. This always returns a fresh
|
||
type. */
|
||
|
||
struct type *
|
||
allocate_stub_method (struct type *type)
|
||
{
|
||
struct type *mtype;
|
||
|
||
mtype = alloc_type_copy (type);
|
||
TYPE_CODE (mtype) = TYPE_CODE_METHOD;
|
||
TYPE_LENGTH (mtype) = 1;
|
||
TYPE_STUB (mtype) = 1;
|
||
TYPE_TARGET_TYPE (mtype) = type;
|
||
/* TYPE_SELF_TYPE (mtype) = unknown yet */
|
||
return mtype;
|
||
}
|
||
|
||
/* Create a range type with a dynamic range from LOW_BOUND to
|
||
HIGH_BOUND, inclusive. See create_range_type for further details. */
|
||
|
||
struct type *
|
||
create_range_type (struct type *result_type, struct type *index_type,
|
||
const struct dynamic_prop *low_bound,
|
||
const struct dynamic_prop *high_bound)
|
||
{
|
||
if (result_type == NULL)
|
||
result_type = alloc_type_copy (index_type);
|
||
TYPE_CODE (result_type) = TYPE_CODE_RANGE;
|
||
TYPE_TARGET_TYPE (result_type) = index_type;
|
||
if (TYPE_STUB (index_type))
|
||
TYPE_TARGET_STUB (result_type) = 1;
|
||
else
|
||
TYPE_LENGTH (result_type) = TYPE_LENGTH (check_typedef (index_type));
|
||
|
||
TYPE_RANGE_DATA (result_type) = (struct range_bounds *)
|
||
TYPE_ZALLOC (result_type, sizeof (struct range_bounds));
|
||
TYPE_RANGE_DATA (result_type)->low = *low_bound;
|
||
TYPE_RANGE_DATA (result_type)->high = *high_bound;
|
||
|
||
if (low_bound->kind == PROP_CONST && low_bound->data.const_val >= 0)
|
||
TYPE_UNSIGNED (result_type) = 1;
|
||
|
||
/* Ada allows the declaration of range types whose upper bound is
|
||
less than the lower bound, so checking the lower bound is not
|
||
enough. Make sure we do not mark a range type whose upper bound
|
||
is negative as unsigned. */
|
||
if (high_bound->kind == PROP_CONST && high_bound->data.const_val < 0)
|
||
TYPE_UNSIGNED (result_type) = 0;
|
||
|
||
return result_type;
|
||
}
|
||
|
||
/* Create a range type using either a blank type supplied in
|
||
RESULT_TYPE, or creating a new type, inheriting the objfile from
|
||
INDEX_TYPE.
|
||
|
||
Indices will be of type INDEX_TYPE, and will range from LOW_BOUND
|
||
to HIGH_BOUND, inclusive.
|
||
|
||
FIXME: Maybe we should check the TYPE_CODE of RESULT_TYPE to make
|
||
sure it is TYPE_CODE_UNDEF before we bash it into a range type? */
|
||
|
||
struct type *
|
||
create_static_range_type (struct type *result_type, struct type *index_type,
|
||
LONGEST low_bound, LONGEST high_bound)
|
||
{
|
||
struct dynamic_prop low, high;
|
||
|
||
low.kind = PROP_CONST;
|
||
low.data.const_val = low_bound;
|
||
|
||
high.kind = PROP_CONST;
|
||
high.data.const_val = high_bound;
|
||
|
||
result_type = create_range_type (result_type, index_type, &low, &high);
|
||
|
||
return result_type;
|
||
}
|
||
|
||
/* Predicate tests whether BOUNDS are static. Returns 1 if all bounds values
|
||
are static, otherwise returns 0. */
|
||
|
||
static int
|
||
has_static_range (const struct range_bounds *bounds)
|
||
{
|
||
return (bounds->low.kind == PROP_CONST
|
||
&& bounds->high.kind == PROP_CONST);
|
||
}
|
||
|
||
|
||
/* Set *LOWP and *HIGHP to the lower and upper bounds of discrete type
|
||
TYPE. Return 1 if type is a range type, 0 if it is discrete (and
|
||
bounds will fit in LONGEST), or -1 otherwise. */
|
||
|
||
int
|
||
get_discrete_bounds (struct type *type, LONGEST *lowp, LONGEST *highp)
|
||
{
|
||
type = check_typedef (type);
|
||
switch (TYPE_CODE (type))
|
||
{
|
||
case TYPE_CODE_RANGE:
|
||
*lowp = TYPE_LOW_BOUND (type);
|
||
*highp = TYPE_HIGH_BOUND (type);
|
||
return 1;
|
||
case TYPE_CODE_ENUM:
|
||
if (TYPE_NFIELDS (type) > 0)
|
||
{
|
||
/* The enums may not be sorted by value, so search all
|
||
entries. */
|
||
int i;
|
||
|
||
*lowp = *highp = TYPE_FIELD_ENUMVAL (type, 0);
|
||
for (i = 0; i < TYPE_NFIELDS (type); i++)
|
||
{
|
||
if (TYPE_FIELD_ENUMVAL (type, i) < *lowp)
|
||
*lowp = TYPE_FIELD_ENUMVAL (type, i);
|
||
if (TYPE_FIELD_ENUMVAL (type, i) > *highp)
|
||
*highp = TYPE_FIELD_ENUMVAL (type, i);
|
||
}
|
||
|
||
/* Set unsigned indicator if warranted. */
|
||
if (*lowp >= 0)
|
||
{
|
||
TYPE_UNSIGNED (type) = 1;
|
||
}
|
||
}
|
||
else
|
||
{
|
||
*lowp = 0;
|
||
*highp = -1;
|
||
}
|
||
return 0;
|
||
case TYPE_CODE_BOOL:
|
||
*lowp = 0;
|
||
*highp = 1;
|
||
return 0;
|
||
case TYPE_CODE_INT:
|
||
if (TYPE_LENGTH (type) > sizeof (LONGEST)) /* Too big */
|
||
return -1;
|
||
if (!TYPE_UNSIGNED (type))
|
||
{
|
||
*lowp = -(1 << (TYPE_LENGTH (type) * TARGET_CHAR_BIT - 1));
|
||
*highp = -*lowp - 1;
|
||
return 0;
|
||
}
|
||
/* ... fall through for unsigned ints ... */
|
||
case TYPE_CODE_CHAR:
|
||
*lowp = 0;
|
||
/* This round-about calculation is to avoid shifting by
|
||
TYPE_LENGTH (type) * TARGET_CHAR_BIT, which will not work
|
||
if TYPE_LENGTH (type) == sizeof (LONGEST). */
|
||
*highp = 1 << (TYPE_LENGTH (type) * TARGET_CHAR_BIT - 1);
|
||
*highp = (*highp - 1) | *highp;
|
||
return 0;
|
||
default:
|
||
return -1;
|
||
}
|
||
}
|
||
|
||
/* Assuming TYPE is a simple, non-empty array type, compute its upper
|
||
and lower bound. Save the low bound into LOW_BOUND if not NULL.
|
||
Save the high bound into HIGH_BOUND if not NULL.
|
||
|
||
Return 1 if the operation was successful. Return zero otherwise,
|
||
in which case the values of LOW_BOUND and HIGH_BOUNDS are unmodified.
|
||
|
||
We now simply use get_discrete_bounds call to get the values
|
||
of the low and high bounds.
|
||
get_discrete_bounds can return three values:
|
||
1, meaning that index is a range,
|
||
0, meaning that index is a discrete type,
|
||
or -1 for failure. */
|
||
|
||
int
|
||
get_array_bounds (struct type *type, LONGEST *low_bound, LONGEST *high_bound)
|
||
{
|
||
struct type *index = TYPE_INDEX_TYPE (type);
|
||
LONGEST low = 0;
|
||
LONGEST high = 0;
|
||
int res;
|
||
|
||
if (index == NULL)
|
||
return 0;
|
||
|
||
res = get_discrete_bounds (index, &low, &high);
|
||
if (res == -1)
|
||
return 0;
|
||
|
||
/* Check if the array bounds are undefined. */
|
||
if (res == 1
|
||
&& ((low_bound && TYPE_ARRAY_LOWER_BOUND_IS_UNDEFINED (type))
|
||
|| (high_bound && TYPE_ARRAY_UPPER_BOUND_IS_UNDEFINED (type))))
|
||
return 0;
|
||
|
||
if (low_bound)
|
||
*low_bound = low;
|
||
|
||
if (high_bound)
|
||
*high_bound = high;
|
||
|
||
return 1;
|
||
}
|
||
|
||
/* Assuming that TYPE is a discrete type and VAL is a valid integer
|
||
representation of a value of this type, save the corresponding
|
||
position number in POS.
|
||
|
||
Its differs from VAL only in the case of enumeration types. In
|
||
this case, the position number of the value of the first listed
|
||
enumeration literal is zero; the position number of the value of
|
||
each subsequent enumeration literal is one more than that of its
|
||
predecessor in the list.
|
||
|
||
Return 1 if the operation was successful. Return zero otherwise,
|
||
in which case the value of POS is unmodified.
|
||
*/
|
||
|
||
int
|
||
discrete_position (struct type *type, LONGEST val, LONGEST *pos)
|
||
{
|
||
if (TYPE_CODE (type) == TYPE_CODE_ENUM)
|
||
{
|
||
int i;
|
||
|
||
for (i = 0; i < TYPE_NFIELDS (type); i += 1)
|
||
{
|
||
if (val == TYPE_FIELD_ENUMVAL (type, i))
|
||
{
|
||
*pos = i;
|
||
return 1;
|
||
}
|
||
}
|
||
/* Invalid enumeration value. */
|
||
return 0;
|
||
}
|
||
else
|
||
{
|
||
*pos = val;
|
||
return 1;
|
||
}
|
||
}
|
||
|
||
/* Create an array type using either a blank type supplied in
|
||
RESULT_TYPE, or creating a new type, inheriting the objfile from
|
||
RANGE_TYPE.
|
||
|
||
Elements will be of type ELEMENT_TYPE, the indices will be of type
|
||
RANGE_TYPE.
|
||
|
||
If BIT_STRIDE is not zero, build a packed array type whose element
|
||
size is BIT_STRIDE. Otherwise, ignore this parameter.
|
||
|
||
FIXME: Maybe we should check the TYPE_CODE of RESULT_TYPE to make
|
||
sure it is TYPE_CODE_UNDEF before we bash it into an array
|
||
type? */
|
||
|
||
struct type *
|
||
create_array_type_with_stride (struct type *result_type,
|
||
struct type *element_type,
|
||
struct type *range_type,
|
||
unsigned int bit_stride)
|
||
{
|
||
if (result_type == NULL)
|
||
result_type = alloc_type_copy (range_type);
|
||
|
||
TYPE_CODE (result_type) = TYPE_CODE_ARRAY;
|
||
TYPE_TARGET_TYPE (result_type) = element_type;
|
||
if (has_static_range (TYPE_RANGE_DATA (range_type)))
|
||
{
|
||
LONGEST low_bound, high_bound;
|
||
|
||
if (get_discrete_bounds (range_type, &low_bound, &high_bound) < 0)
|
||
low_bound = high_bound = 0;
|
||
element_type = check_typedef (element_type);
|
||
/* Be careful when setting the array length. Ada arrays can be
|
||
empty arrays with the high_bound being smaller than the low_bound.
|
||
In such cases, the array length should be zero. */
|
||
if (high_bound < low_bound)
|
||
TYPE_LENGTH (result_type) = 0;
|
||
else if (bit_stride > 0)
|
||
TYPE_LENGTH (result_type) =
|
||
(bit_stride * (high_bound - low_bound + 1) + 7) / 8;
|
||
else
|
||
TYPE_LENGTH (result_type) =
|
||
TYPE_LENGTH (element_type) * (high_bound - low_bound + 1);
|
||
}
|
||
else
|
||
{
|
||
/* This type is dynamic and its length needs to be computed
|
||
on demand. In the meantime, avoid leaving the TYPE_LENGTH
|
||
undefined by setting it to zero. Although we are not expected
|
||
to trust TYPE_LENGTH in this case, setting the size to zero
|
||
allows us to avoid allocating objects of random sizes in case
|
||
we accidently do. */
|
||
TYPE_LENGTH (result_type) = 0;
|
||
}
|
||
|
||
TYPE_NFIELDS (result_type) = 1;
|
||
TYPE_FIELDS (result_type) =
|
||
(struct field *) TYPE_ZALLOC (result_type, sizeof (struct field));
|
||
TYPE_INDEX_TYPE (result_type) = range_type;
|
||
if (bit_stride > 0)
|
||
TYPE_FIELD_BITSIZE (result_type, 0) = bit_stride;
|
||
|
||
/* TYPE_FLAG_TARGET_STUB will take care of zero length arrays. */
|
||
if (TYPE_LENGTH (result_type) == 0)
|
||
TYPE_TARGET_STUB (result_type) = 1;
|
||
|
||
return result_type;
|
||
}
|
||
|
||
/* Same as create_array_type_with_stride but with no bit_stride
|
||
(BIT_STRIDE = 0), thus building an unpacked array. */
|
||
|
||
struct type *
|
||
create_array_type (struct type *result_type,
|
||
struct type *element_type,
|
||
struct type *range_type)
|
||
{
|
||
return create_array_type_with_stride (result_type, element_type,
|
||
range_type, 0);
|
||
}
|
||
|
||
struct type *
|
||
lookup_array_range_type (struct type *element_type,
|
||
LONGEST low_bound, LONGEST high_bound)
|
||
{
|
||
struct gdbarch *gdbarch = get_type_arch (element_type);
|
||
struct type *index_type = builtin_type (gdbarch)->builtin_int;
|
||
struct type *range_type
|
||
= create_static_range_type (NULL, index_type, low_bound, high_bound);
|
||
|
||
return create_array_type (NULL, element_type, range_type);
|
||
}
|
||
|
||
/* Create a string type using either a blank type supplied in
|
||
RESULT_TYPE, or creating a new type. String types are similar
|
||
enough to array of char types that we can use create_array_type to
|
||
build the basic type and then bash it into a string type.
|
||
|
||
For fixed length strings, the range type contains 0 as the lower
|
||
bound and the length of the string minus one as the upper bound.
|
||
|
||
FIXME: Maybe we should check the TYPE_CODE of RESULT_TYPE to make
|
||
sure it is TYPE_CODE_UNDEF before we bash it into a string
|
||
type? */
|
||
|
||
struct type *
|
||
create_string_type (struct type *result_type,
|
||
struct type *string_char_type,
|
||
struct type *range_type)
|
||
{
|
||
result_type = create_array_type (result_type,
|
||
string_char_type,
|
||
range_type);
|
||
TYPE_CODE (result_type) = TYPE_CODE_STRING;
|
||
return result_type;
|
||
}
|
||
|
||
struct type *
|
||
lookup_string_range_type (struct type *string_char_type,
|
||
LONGEST low_bound, LONGEST high_bound)
|
||
{
|
||
struct type *result_type;
|
||
|
||
result_type = lookup_array_range_type (string_char_type,
|
||
low_bound, high_bound);
|
||
TYPE_CODE (result_type) = TYPE_CODE_STRING;
|
||
return result_type;
|
||
}
|
||
|
||
struct type *
|
||
create_set_type (struct type *result_type, struct type *domain_type)
|
||
{
|
||
if (result_type == NULL)
|
||
result_type = alloc_type_copy (domain_type);
|
||
|
||
TYPE_CODE (result_type) = TYPE_CODE_SET;
|
||
TYPE_NFIELDS (result_type) = 1;
|
||
TYPE_FIELDS (result_type)
|
||
= (struct field *) TYPE_ZALLOC (result_type, sizeof (struct field));
|
||
|
||
if (!TYPE_STUB (domain_type))
|
||
{
|
||
LONGEST low_bound, high_bound, bit_length;
|
||
|
||
if (get_discrete_bounds (domain_type, &low_bound, &high_bound) < 0)
|
||
low_bound = high_bound = 0;
|
||
bit_length = high_bound - low_bound + 1;
|
||
TYPE_LENGTH (result_type)
|
||
= (bit_length + TARGET_CHAR_BIT - 1) / TARGET_CHAR_BIT;
|
||
if (low_bound >= 0)
|
||
TYPE_UNSIGNED (result_type) = 1;
|
||
}
|
||
TYPE_FIELD_TYPE (result_type, 0) = domain_type;
|
||
|
||
return result_type;
|
||
}
|
||
|
||
/* Convert ARRAY_TYPE to a vector type. This may modify ARRAY_TYPE
|
||
and any array types nested inside it. */
|
||
|
||
void
|
||
make_vector_type (struct type *array_type)
|
||
{
|
||
struct type *inner_array, *elt_type;
|
||
int flags;
|
||
|
||
/* Find the innermost array type, in case the array is
|
||
multi-dimensional. */
|
||
inner_array = array_type;
|
||
while (TYPE_CODE (TYPE_TARGET_TYPE (inner_array)) == TYPE_CODE_ARRAY)
|
||
inner_array = TYPE_TARGET_TYPE (inner_array);
|
||
|
||
elt_type = TYPE_TARGET_TYPE (inner_array);
|
||
if (TYPE_CODE (elt_type) == TYPE_CODE_INT)
|
||
{
|
||
flags = TYPE_INSTANCE_FLAGS (elt_type) | TYPE_INSTANCE_FLAG_NOTTEXT;
|
||
elt_type = make_qualified_type (elt_type, flags, NULL);
|
||
TYPE_TARGET_TYPE (inner_array) = elt_type;
|
||
}
|
||
|
||
TYPE_VECTOR (array_type) = 1;
|
||
}
|
||
|
||
struct type *
|
||
init_vector_type (struct type *elt_type, int n)
|
||
{
|
||
struct type *array_type;
|
||
|
||
array_type = lookup_array_range_type (elt_type, 0, n - 1);
|
||
make_vector_type (array_type);
|
||
return array_type;
|
||
}
|
||
|
||
/* Internal routine called by TYPE_SELF_TYPE to return the type that TYPE
|
||
belongs to. In c++ this is the class of "this", but TYPE_THIS_TYPE is too
|
||
confusing. "self" is a common enough replacement for "this".
|
||
TYPE must be one of TYPE_CODE_METHODPTR, TYPE_CODE_MEMBERPTR, or
|
||
TYPE_CODE_METHOD. */
|
||
|
||
struct type *
|
||
internal_type_self_type (struct type *type)
|
||
{
|
||
switch (TYPE_CODE (type))
|
||
{
|
||
case TYPE_CODE_METHODPTR:
|
||
case TYPE_CODE_MEMBERPTR:
|
||
if (TYPE_SPECIFIC_FIELD (type) == TYPE_SPECIFIC_NONE)
|
||
return NULL;
|
||
gdb_assert (TYPE_SPECIFIC_FIELD (type) == TYPE_SPECIFIC_SELF_TYPE);
|
||
return TYPE_MAIN_TYPE (type)->type_specific.self_type;
|
||
case TYPE_CODE_METHOD:
|
||
if (TYPE_SPECIFIC_FIELD (type) == TYPE_SPECIFIC_NONE)
|
||
return NULL;
|
||
gdb_assert (TYPE_SPECIFIC_FIELD (type) == TYPE_SPECIFIC_FUNC);
|
||
return TYPE_MAIN_TYPE (type)->type_specific.func_stuff->self_type;
|
||
default:
|
||
gdb_assert_not_reached ("bad type");
|
||
}
|
||
}
|
||
|
||
/* Set the type of the class that TYPE belongs to.
|
||
In c++ this is the class of "this".
|
||
TYPE must be one of TYPE_CODE_METHODPTR, TYPE_CODE_MEMBERPTR, or
|
||
TYPE_CODE_METHOD. */
|
||
|
||
void
|
||
set_type_self_type (struct type *type, struct type *self_type)
|
||
{
|
||
switch (TYPE_CODE (type))
|
||
{
|
||
case TYPE_CODE_METHODPTR:
|
||
case TYPE_CODE_MEMBERPTR:
|
||
if (TYPE_SPECIFIC_FIELD (type) == TYPE_SPECIFIC_NONE)
|
||
TYPE_SPECIFIC_FIELD (type) = TYPE_SPECIFIC_SELF_TYPE;
|
||
gdb_assert (TYPE_SPECIFIC_FIELD (type) == TYPE_SPECIFIC_SELF_TYPE);
|
||
TYPE_MAIN_TYPE (type)->type_specific.self_type = self_type;
|
||
break;
|
||
case TYPE_CODE_METHOD:
|
||
if (TYPE_SPECIFIC_FIELD (type) == TYPE_SPECIFIC_NONE)
|
||
INIT_FUNC_SPECIFIC (type);
|
||
gdb_assert (TYPE_SPECIFIC_FIELD (type) == TYPE_SPECIFIC_FUNC);
|
||
TYPE_MAIN_TYPE (type)->type_specific.func_stuff->self_type = self_type;
|
||
break;
|
||
default:
|
||
gdb_assert_not_reached ("bad type");
|
||
}
|
||
}
|
||
|
||
/* Smash TYPE to be a type of pointers to members of SELF_TYPE with type
|
||
TO_TYPE. A member pointer is a wierd thing -- it amounts to a
|
||
typed offset into a struct, e.g. "an int at offset 8". A MEMBER
|
||
TYPE doesn't include the offset (that's the value of the MEMBER
|
||
itself), but does include the structure type into which it points
|
||
(for some reason).
|
||
|
||
When "smashing" the type, we preserve the objfile that the old type
|
||
pointed to, since we aren't changing where the type is actually
|
||
allocated. */
|
||
|
||
void
|
||
smash_to_memberptr_type (struct type *type, struct type *self_type,
|
||
struct type *to_type)
|
||
{
|
||
smash_type (type);
|
||
TYPE_CODE (type) = TYPE_CODE_MEMBERPTR;
|
||
TYPE_TARGET_TYPE (type) = to_type;
|
||
set_type_self_type (type, self_type);
|
||
/* Assume that a data member pointer is the same size as a normal
|
||
pointer. */
|
||
TYPE_LENGTH (type)
|
||
= gdbarch_ptr_bit (get_type_arch (to_type)) / TARGET_CHAR_BIT;
|
||
}
|
||
|
||
/* Smash TYPE to be a type of pointer to methods type TO_TYPE.
|
||
|
||
When "smashing" the type, we preserve the objfile that the old type
|
||
pointed to, since we aren't changing where the type is actually
|
||
allocated. */
|
||
|
||
void
|
||
smash_to_methodptr_type (struct type *type, struct type *to_type)
|
||
{
|
||
smash_type (type);
|
||
TYPE_CODE (type) = TYPE_CODE_METHODPTR;
|
||
TYPE_TARGET_TYPE (type) = to_type;
|
||
set_type_self_type (type, TYPE_SELF_TYPE (to_type));
|
||
TYPE_LENGTH (type) = cplus_method_ptr_size (to_type);
|
||
}
|
||
|
||
/* Smash TYPE to be a type of method of SELF_TYPE with type TO_TYPE.
|
||
METHOD just means `function that gets an extra "this" argument'.
|
||
|
||
When "smashing" the type, we preserve the objfile that the old type
|
||
pointed to, since we aren't changing where the type is actually
|
||
allocated. */
|
||
|
||
void
|
||
smash_to_method_type (struct type *type, struct type *self_type,
|
||
struct type *to_type, struct field *args,
|
||
int nargs, int varargs)
|
||
{
|
||
smash_type (type);
|
||
TYPE_CODE (type) = TYPE_CODE_METHOD;
|
||
TYPE_TARGET_TYPE (type) = to_type;
|
||
set_type_self_type (type, self_type);
|
||
TYPE_FIELDS (type) = args;
|
||
TYPE_NFIELDS (type) = nargs;
|
||
if (varargs)
|
||
TYPE_VARARGS (type) = 1;
|
||
TYPE_LENGTH (type) = 1; /* In practice, this is never needed. */
|
||
}
|
||
|
||
/* Return a typename for a struct/union/enum type without "struct ",
|
||
"union ", or "enum ". If the type has a NULL name, return NULL. */
|
||
|
||
const char *
|
||
type_name_no_tag (const struct type *type)
|
||
{
|
||
if (TYPE_TAG_NAME (type) != NULL)
|
||
return TYPE_TAG_NAME (type);
|
||
|
||
/* Is there code which expects this to return the name if there is
|
||
no tag name? My guess is that this is mainly used for C++ in
|
||
cases where the two will always be the same. */
|
||
return TYPE_NAME (type);
|
||
}
|
||
|
||
/* A wrapper of type_name_no_tag which calls error if the type is anonymous.
|
||
Since GCC PR debug/47510 DWARF provides associated information to detect the
|
||
anonymous class linkage name from its typedef.
|
||
|
||
Parameter TYPE should not yet have CHECK_TYPEDEF applied, this function will
|
||
apply it itself. */
|
||
|
||
const char *
|
||
type_name_no_tag_or_error (struct type *type)
|
||
{
|
||
struct type *saved_type = type;
|
||
const char *name;
|
||
struct objfile *objfile;
|
||
|
||
type = check_typedef (type);
|
||
|
||
name = type_name_no_tag (type);
|
||
if (name != NULL)
|
||
return name;
|
||
|
||
name = type_name_no_tag (saved_type);
|
||
objfile = TYPE_OBJFILE (saved_type);
|
||
error (_("Invalid anonymous type %s [in module %s], GCC PR debug/47510 bug?"),
|
||
name ? name : "<anonymous>",
|
||
objfile ? objfile_name (objfile) : "<arch>");
|
||
}
|
||
|
||
/* Lookup a typedef or primitive type named NAME, visible in lexical
|
||
block BLOCK. If NOERR is nonzero, return zero if NAME is not
|
||
suitably defined. */
|
||
|
||
struct type *
|
||
lookup_typename (const struct language_defn *language,
|
||
struct gdbarch *gdbarch, const char *name,
|
||
const struct block *block, int noerr)
|
||
{
|
||
struct symbol *sym;
|
||
struct type *type;
|
||
|
||
sym = lookup_symbol_in_language (name, block, VAR_DOMAIN,
|
||
language->la_language, NULL).symbol;
|
||
if (sym != NULL && SYMBOL_CLASS (sym) == LOC_TYPEDEF)
|
||
return SYMBOL_TYPE (sym);
|
||
|
||
if (noerr)
|
||
return NULL;
|
||
error (_("No type named %s."), name);
|
||
}
|
||
|
||
struct type *
|
||
lookup_unsigned_typename (const struct language_defn *language,
|
||
struct gdbarch *gdbarch, const char *name)
|
||
{
|
||
char *uns = (char *) alloca (strlen (name) + 10);
|
||
|
||
strcpy (uns, "unsigned ");
|
||
strcpy (uns + 9, name);
|
||
return lookup_typename (language, gdbarch, uns, (struct block *) NULL, 0);
|
||
}
|
||
|
||
struct type *
|
||
lookup_signed_typename (const struct language_defn *language,
|
||
struct gdbarch *gdbarch, const char *name)
|
||
{
|
||
struct type *t;
|
||
char *uns = (char *) alloca (strlen (name) + 8);
|
||
|
||
strcpy (uns, "signed ");
|
||
strcpy (uns + 7, name);
|
||
t = lookup_typename (language, gdbarch, uns, (struct block *) NULL, 1);
|
||
/* If we don't find "signed FOO" just try again with plain "FOO". */
|
||
if (t != NULL)
|
||
return t;
|
||
return lookup_typename (language, gdbarch, name, (struct block *) NULL, 0);
|
||
}
|
||
|
||
/* Lookup a structure type named "struct NAME",
|
||
visible in lexical block BLOCK. */
|
||
|
||
struct type *
|
||
lookup_struct (const char *name, const struct block *block)
|
||
{
|
||
struct symbol *sym;
|
||
|
||
sym = lookup_symbol (name, block, STRUCT_DOMAIN, 0).symbol;
|
||
|
||
if (sym == NULL)
|
||
{
|
||
error (_("No struct type named %s."), name);
|
||
}
|
||
if (TYPE_CODE (SYMBOL_TYPE (sym)) != TYPE_CODE_STRUCT)
|
||
{
|
||
error (_("This context has class, union or enum %s, not a struct."),
|
||
name);
|
||
}
|
||
return (SYMBOL_TYPE (sym));
|
||
}
|
||
|
||
/* Lookup a union type named "union NAME",
|
||
visible in lexical block BLOCK. */
|
||
|
||
struct type *
|
||
lookup_union (const char *name, const struct block *block)
|
||
{
|
||
struct symbol *sym;
|
||
struct type *t;
|
||
|
||
sym = lookup_symbol (name, block, STRUCT_DOMAIN, 0).symbol;
|
||
|
||
if (sym == NULL)
|
||
error (_("No union type named %s."), name);
|
||
|
||
t = SYMBOL_TYPE (sym);
|
||
|
||
if (TYPE_CODE (t) == TYPE_CODE_UNION)
|
||
return t;
|
||
|
||
/* If we get here, it's not a union. */
|
||
error (_("This context has class, struct or enum %s, not a union."),
|
||
name);
|
||
}
|
||
|
||
/* Lookup an enum type named "enum NAME",
|
||
visible in lexical block BLOCK. */
|
||
|
||
struct type *
|
||
lookup_enum (const char *name, const struct block *block)
|
||
{
|
||
struct symbol *sym;
|
||
|
||
sym = lookup_symbol (name, block, STRUCT_DOMAIN, 0).symbol;
|
||
if (sym == NULL)
|
||
{
|
||
error (_("No enum type named %s."), name);
|
||
}
|
||
if (TYPE_CODE (SYMBOL_TYPE (sym)) != TYPE_CODE_ENUM)
|
||
{
|
||
error (_("This context has class, struct or union %s, not an enum."),
|
||
name);
|
||
}
|
||
return (SYMBOL_TYPE (sym));
|
||
}
|
||
|
||
/* Lookup a template type named "template NAME<TYPE>",
|
||
visible in lexical block BLOCK. */
|
||
|
||
struct type *
|
||
lookup_template_type (char *name, struct type *type,
|
||
const struct block *block)
|
||
{
|
||
struct symbol *sym;
|
||
char *nam = (char *)
|
||
alloca (strlen (name) + strlen (TYPE_NAME (type)) + 4);
|
||
|
||
strcpy (nam, name);
|
||
strcat (nam, "<");
|
||
strcat (nam, TYPE_NAME (type));
|
||
strcat (nam, " >"); /* FIXME, extra space still introduced in gcc? */
|
||
|
||
sym = lookup_symbol (nam, block, VAR_DOMAIN, 0).symbol;
|
||
|
||
if (sym == NULL)
|
||
{
|
||
error (_("No template type named %s."), name);
|
||
}
|
||
if (TYPE_CODE (SYMBOL_TYPE (sym)) != TYPE_CODE_STRUCT)
|
||
{
|
||
error (_("This context has class, union or enum %s, not a struct."),
|
||
name);
|
||
}
|
||
return (SYMBOL_TYPE (sym));
|
||
}
|
||
|
||
/* Given a type TYPE, lookup the type of the component of type named
|
||
NAME.
|
||
|
||
TYPE can be either a struct or union, or a pointer or reference to
|
||
a struct or union. If it is a pointer or reference, its target
|
||
type is automatically used. Thus '.' and '->' are interchangable,
|
||
as specified for the definitions of the expression element types
|
||
STRUCTOP_STRUCT and STRUCTOP_PTR.
|
||
|
||
If NOERR is nonzero, return zero if NAME is not suitably defined.
|
||
If NAME is the name of a baseclass type, return that type. */
|
||
|
||
struct type *
|
||
lookup_struct_elt_type (struct type *type, const char *name, int noerr)
|
||
{
|
||
int i;
|
||
char *type_name;
|
||
|
||
for (;;)
|
||
{
|
||
type = check_typedef (type);
|
||
if (TYPE_CODE (type) != TYPE_CODE_PTR
|
||
&& TYPE_CODE (type) != TYPE_CODE_REF)
|
||
break;
|
||
type = TYPE_TARGET_TYPE (type);
|
||
}
|
||
|
||
if (TYPE_CODE (type) != TYPE_CODE_STRUCT
|
||
&& TYPE_CODE (type) != TYPE_CODE_UNION)
|
||
{
|
||
type_name = type_to_string (type);
|
||
make_cleanup (xfree, type_name);
|
||
error (_("Type %s is not a structure or union type."), type_name);
|
||
}
|
||
|
||
#if 0
|
||
/* FIXME: This change put in by Michael seems incorrect for the case
|
||
where the structure tag name is the same as the member name.
|
||
I.e. when doing "ptype bell->bar" for "struct foo { int bar; int
|
||
foo; } bell;" Disabled by fnf. */
|
||
{
|
||
char *type_name;
|
||
|
||
type_name = type_name_no_tag (type);
|
||
if (type_name != NULL && strcmp (type_name, name) == 0)
|
||
return type;
|
||
}
|
||
#endif
|
||
|
||
for (i = TYPE_NFIELDS (type) - 1; i >= TYPE_N_BASECLASSES (type); i--)
|
||
{
|
||
const char *t_field_name = TYPE_FIELD_NAME (type, i);
|
||
|
||
if (t_field_name && (strcmp_iw (t_field_name, name) == 0))
|
||
{
|
||
return TYPE_FIELD_TYPE (type, i);
|
||
}
|
||
else if (!t_field_name || *t_field_name == '\0')
|
||
{
|
||
struct type *subtype
|
||
= lookup_struct_elt_type (TYPE_FIELD_TYPE (type, i), name, 1);
|
||
|
||
if (subtype != NULL)
|
||
return subtype;
|
||
}
|
||
}
|
||
|
||
/* OK, it's not in this class. Recursively check the baseclasses. */
|
||
for (i = TYPE_N_BASECLASSES (type) - 1; i >= 0; i--)
|
||
{
|
||
struct type *t;
|
||
|
||
t = lookup_struct_elt_type (TYPE_BASECLASS (type, i), name, 1);
|
||
if (t != NULL)
|
||
{
|
||
return t;
|
||
}
|
||
}
|
||
|
||
if (noerr)
|
||
{
|
||
return NULL;
|
||
}
|
||
|
||
type_name = type_to_string (type);
|
||
make_cleanup (xfree, type_name);
|
||
error (_("Type %s has no component named %s."), type_name, name);
|
||
}
|
||
|
||
/* Store in *MAX the largest number representable by unsigned integer type
|
||
TYPE. */
|
||
|
||
void
|
||
get_unsigned_type_max (struct type *type, ULONGEST *max)
|
||
{
|
||
unsigned int n;
|
||
|
||
type = check_typedef (type);
|
||
gdb_assert (TYPE_CODE (type) == TYPE_CODE_INT && TYPE_UNSIGNED (type));
|
||
gdb_assert (TYPE_LENGTH (type) <= sizeof (ULONGEST));
|
||
|
||
/* Written this way to avoid overflow. */
|
||
n = TYPE_LENGTH (type) * TARGET_CHAR_BIT;
|
||
*max = ((((ULONGEST) 1 << (n - 1)) - 1) << 1) | 1;
|
||
}
|
||
|
||
/* Store in *MIN, *MAX the smallest and largest numbers representable by
|
||
signed integer type TYPE. */
|
||
|
||
void
|
||
get_signed_type_minmax (struct type *type, LONGEST *min, LONGEST *max)
|
||
{
|
||
unsigned int n;
|
||
|
||
type = check_typedef (type);
|
||
gdb_assert (TYPE_CODE (type) == TYPE_CODE_INT && !TYPE_UNSIGNED (type));
|
||
gdb_assert (TYPE_LENGTH (type) <= sizeof (LONGEST));
|
||
|
||
n = TYPE_LENGTH (type) * TARGET_CHAR_BIT;
|
||
*min = -((ULONGEST) 1 << (n - 1));
|
||
*max = ((ULONGEST) 1 << (n - 1)) - 1;
|
||
}
|
||
|
||
/* Internal routine called by TYPE_VPTR_FIELDNO to return the value of
|
||
cplus_stuff.vptr_fieldno.
|
||
|
||
cplus_stuff is initialized to cplus_struct_default which does not
|
||
set vptr_fieldno to -1 for portability reasons (IWBN to use C99
|
||
designated initializers). We cope with that here. */
|
||
|
||
int
|
||
internal_type_vptr_fieldno (struct type *type)
|
||
{
|
||
type = check_typedef (type);
|
||
gdb_assert (TYPE_CODE (type) == TYPE_CODE_STRUCT
|
||
|| TYPE_CODE (type) == TYPE_CODE_UNION);
|
||
if (!HAVE_CPLUS_STRUCT (type))
|
||
return -1;
|
||
return TYPE_RAW_CPLUS_SPECIFIC (type)->vptr_fieldno;
|
||
}
|
||
|
||
/* Set the value of cplus_stuff.vptr_fieldno. */
|
||
|
||
void
|
||
set_type_vptr_fieldno (struct type *type, int fieldno)
|
||
{
|
||
type = check_typedef (type);
|
||
gdb_assert (TYPE_CODE (type) == TYPE_CODE_STRUCT
|
||
|| TYPE_CODE (type) == TYPE_CODE_UNION);
|
||
if (!HAVE_CPLUS_STRUCT (type))
|
||
ALLOCATE_CPLUS_STRUCT_TYPE (type);
|
||
TYPE_RAW_CPLUS_SPECIFIC (type)->vptr_fieldno = fieldno;
|
||
}
|
||
|
||
/* Internal routine called by TYPE_VPTR_BASETYPE to return the value of
|
||
cplus_stuff.vptr_basetype. */
|
||
|
||
struct type *
|
||
internal_type_vptr_basetype (struct type *type)
|
||
{
|
||
type = check_typedef (type);
|
||
gdb_assert (TYPE_CODE (type) == TYPE_CODE_STRUCT
|
||
|| TYPE_CODE (type) == TYPE_CODE_UNION);
|
||
gdb_assert (TYPE_SPECIFIC_FIELD (type) == TYPE_SPECIFIC_CPLUS_STUFF);
|
||
return TYPE_RAW_CPLUS_SPECIFIC (type)->vptr_basetype;
|
||
}
|
||
|
||
/* Set the value of cplus_stuff.vptr_basetype. */
|
||
|
||
void
|
||
set_type_vptr_basetype (struct type *type, struct type *basetype)
|
||
{
|
||
type = check_typedef (type);
|
||
gdb_assert (TYPE_CODE (type) == TYPE_CODE_STRUCT
|
||
|| TYPE_CODE (type) == TYPE_CODE_UNION);
|
||
if (!HAVE_CPLUS_STRUCT (type))
|
||
ALLOCATE_CPLUS_STRUCT_TYPE (type);
|
||
TYPE_RAW_CPLUS_SPECIFIC (type)->vptr_basetype = basetype;
|
||
}
|
||
|
||
/* Lookup the vptr basetype/fieldno values for TYPE.
|
||
If found store vptr_basetype in *BASETYPEP if non-NULL, and return
|
||
vptr_fieldno. Also, if found and basetype is from the same objfile,
|
||
cache the results.
|
||
If not found, return -1 and ignore BASETYPEP.
|
||
Callers should be aware that in some cases (for example,
|
||
the type or one of its baseclasses is a stub type and we are
|
||
debugging a .o file, or the compiler uses DWARF-2 and is not GCC),
|
||
this function will not be able to find the
|
||
virtual function table pointer, and vptr_fieldno will remain -1 and
|
||
vptr_basetype will remain NULL or incomplete. */
|
||
|
||
int
|
||
get_vptr_fieldno (struct type *type, struct type **basetypep)
|
||
{
|
||
type = check_typedef (type);
|
||
|
||
if (TYPE_VPTR_FIELDNO (type) < 0)
|
||
{
|
||
int i;
|
||
|
||
/* We must start at zero in case the first (and only) baseclass
|
||
is virtual (and hence we cannot share the table pointer). */
|
||
for (i = 0; i < TYPE_N_BASECLASSES (type); i++)
|
||
{
|
||
struct type *baseclass = check_typedef (TYPE_BASECLASS (type, i));
|
||
int fieldno;
|
||
struct type *basetype;
|
||
|
||
fieldno = get_vptr_fieldno (baseclass, &basetype);
|
||
if (fieldno >= 0)
|
||
{
|
||
/* If the type comes from a different objfile we can't cache
|
||
it, it may have a different lifetime. PR 2384 */
|
||
if (TYPE_OBJFILE (type) == TYPE_OBJFILE (basetype))
|
||
{
|
||
set_type_vptr_fieldno (type, fieldno);
|
||
set_type_vptr_basetype (type, basetype);
|
||
}
|
||
if (basetypep)
|
||
*basetypep = basetype;
|
||
return fieldno;
|
||
}
|
||
}
|
||
|
||
/* Not found. */
|
||
return -1;
|
||
}
|
||
else
|
||
{
|
||
if (basetypep)
|
||
*basetypep = TYPE_VPTR_BASETYPE (type);
|
||
return TYPE_VPTR_FIELDNO (type);
|
||
}
|
||
}
|
||
|
||
static void
|
||
stub_noname_complaint (void)
|
||
{
|
||
complaint (&symfile_complaints, _("stub type has NULL name"));
|
||
}
|
||
|
||
/* Worker for is_dynamic_type. */
|
||
|
||
static int
|
||
is_dynamic_type_internal (struct type *type, int top_level)
|
||
{
|
||
type = check_typedef (type);
|
||
|
||
/* We only want to recognize references at the outermost level. */
|
||
if (top_level && TYPE_CODE (type) == TYPE_CODE_REF)
|
||
type = check_typedef (TYPE_TARGET_TYPE (type));
|
||
|
||
/* Types that have a dynamic TYPE_DATA_LOCATION are considered
|
||
dynamic, even if the type itself is statically defined.
|
||
From a user's point of view, this may appear counter-intuitive;
|
||
but it makes sense in this context, because the point is to determine
|
||
whether any part of the type needs to be resolved before it can
|
||
be exploited. */
|
||
if (TYPE_DATA_LOCATION (type) != NULL
|
||
&& (TYPE_DATA_LOCATION_KIND (type) == PROP_LOCEXPR
|
||
|| TYPE_DATA_LOCATION_KIND (type) == PROP_LOCLIST))
|
||
return 1;
|
||
|
||
switch (TYPE_CODE (type))
|
||
{
|
||
case TYPE_CODE_RANGE:
|
||
{
|
||
/* A range type is obviously dynamic if it has at least one
|
||
dynamic bound. But also consider the range type to be
|
||
dynamic when its subtype is dynamic, even if the bounds
|
||
of the range type are static. It allows us to assume that
|
||
the subtype of a static range type is also static. */
|
||
return (!has_static_range (TYPE_RANGE_DATA (type))
|
||
|| is_dynamic_type_internal (TYPE_TARGET_TYPE (type), 0));
|
||
}
|
||
|
||
case TYPE_CODE_ARRAY:
|
||
{
|
||
gdb_assert (TYPE_NFIELDS (type) == 1);
|
||
|
||
/* The array is dynamic if either the bounds are dynamic,
|
||
or the elements it contains have a dynamic contents. */
|
||
if (is_dynamic_type_internal (TYPE_INDEX_TYPE (type), 0))
|
||
return 1;
|
||
return is_dynamic_type_internal (TYPE_TARGET_TYPE (type), 0);
|
||
}
|
||
|
||
case TYPE_CODE_STRUCT:
|
||
case TYPE_CODE_UNION:
|
||
{
|
||
int i;
|
||
|
||
for (i = 0; i < TYPE_NFIELDS (type); ++i)
|
||
if (!field_is_static (&TYPE_FIELD (type, i))
|
||
&& is_dynamic_type_internal (TYPE_FIELD_TYPE (type, i), 0))
|
||
return 1;
|
||
}
|
||
break;
|
||
}
|
||
|
||
return 0;
|
||
}
|
||
|
||
/* See gdbtypes.h. */
|
||
|
||
int
|
||
is_dynamic_type (struct type *type)
|
||
{
|
||
return is_dynamic_type_internal (type, 1);
|
||
}
|
||
|
||
static struct type *resolve_dynamic_type_internal
|
||
(struct type *type, struct property_addr_info *addr_stack, int top_level);
|
||
|
||
/* Given a dynamic range type (dyn_range_type) and a stack of
|
||
struct property_addr_info elements, return a static version
|
||
of that type. */
|
||
|
||
static struct type *
|
||
resolve_dynamic_range (struct type *dyn_range_type,
|
||
struct property_addr_info *addr_stack)
|
||
{
|
||
CORE_ADDR value;
|
||
struct type *static_range_type, *static_target_type;
|
||
const struct dynamic_prop *prop;
|
||
const struct dwarf2_locexpr_baton *baton;
|
||
struct dynamic_prop low_bound, high_bound;
|
||
|
||
gdb_assert (TYPE_CODE (dyn_range_type) == TYPE_CODE_RANGE);
|
||
|
||
prop = &TYPE_RANGE_DATA (dyn_range_type)->low;
|
||
if (dwarf2_evaluate_property (prop, NULL, addr_stack, &value))
|
||
{
|
||
low_bound.kind = PROP_CONST;
|
||
low_bound.data.const_val = value;
|
||
}
|
||
else
|
||
{
|
||
low_bound.kind = PROP_UNDEFINED;
|
||
low_bound.data.const_val = 0;
|
||
}
|
||
|
||
prop = &TYPE_RANGE_DATA (dyn_range_type)->high;
|
||
if (dwarf2_evaluate_property (prop, NULL, addr_stack, &value))
|
||
{
|
||
high_bound.kind = PROP_CONST;
|
||
high_bound.data.const_val = value;
|
||
|
||
if (TYPE_RANGE_DATA (dyn_range_type)->flag_upper_bound_is_count)
|
||
high_bound.data.const_val
|
||
= low_bound.data.const_val + high_bound.data.const_val - 1;
|
||
}
|
||
else
|
||
{
|
||
high_bound.kind = PROP_UNDEFINED;
|
||
high_bound.data.const_val = 0;
|
||
}
|
||
|
||
static_target_type
|
||
= resolve_dynamic_type_internal (TYPE_TARGET_TYPE (dyn_range_type),
|
||
addr_stack, 0);
|
||
static_range_type = create_range_type (copy_type (dyn_range_type),
|
||
static_target_type,
|
||
&low_bound, &high_bound);
|
||
TYPE_RANGE_DATA (static_range_type)->flag_bound_evaluated = 1;
|
||
return static_range_type;
|
||
}
|
||
|
||
/* Resolves dynamic bound values of an array type TYPE to static ones.
|
||
ADDR_STACK is a stack of struct property_addr_info to be used
|
||
if needed during the dynamic resolution. */
|
||
|
||
static struct type *
|
||
resolve_dynamic_array (struct type *type,
|
||
struct property_addr_info *addr_stack)
|
||
{
|
||
CORE_ADDR value;
|
||
struct type *elt_type;
|
||
struct type *range_type;
|
||
struct type *ary_dim;
|
||
|
||
gdb_assert (TYPE_CODE (type) == TYPE_CODE_ARRAY);
|
||
|
||
elt_type = type;
|
||
range_type = check_typedef (TYPE_INDEX_TYPE (elt_type));
|
||
range_type = resolve_dynamic_range (range_type, addr_stack);
|
||
|
||
ary_dim = check_typedef (TYPE_TARGET_TYPE (elt_type));
|
||
|
||
if (ary_dim != NULL && TYPE_CODE (ary_dim) == TYPE_CODE_ARRAY)
|
||
elt_type = resolve_dynamic_array (ary_dim, addr_stack);
|
||
else
|
||
elt_type = TYPE_TARGET_TYPE (type);
|
||
|
||
return create_array_type_with_stride (copy_type (type),
|
||
elt_type, range_type,
|
||
TYPE_FIELD_BITSIZE (type, 0));
|
||
}
|
||
|
||
/* Resolve dynamic bounds of members of the union TYPE to static
|
||
bounds. ADDR_STACK is a stack of struct property_addr_info
|
||
to be used if needed during the dynamic resolution. */
|
||
|
||
static struct type *
|
||
resolve_dynamic_union (struct type *type,
|
||
struct property_addr_info *addr_stack)
|
||
{
|
||
struct type *resolved_type;
|
||
int i;
|
||
unsigned int max_len = 0;
|
||
|
||
gdb_assert (TYPE_CODE (type) == TYPE_CODE_UNION);
|
||
|
||
resolved_type = copy_type (type);
|
||
TYPE_FIELDS (resolved_type)
|
||
= (struct field *) TYPE_ALLOC (resolved_type,
|
||
TYPE_NFIELDS (resolved_type)
|
||
* sizeof (struct field));
|
||
memcpy (TYPE_FIELDS (resolved_type),
|
||
TYPE_FIELDS (type),
|
||
TYPE_NFIELDS (resolved_type) * sizeof (struct field));
|
||
for (i = 0; i < TYPE_NFIELDS (resolved_type); ++i)
|
||
{
|
||
struct type *t;
|
||
|
||
if (field_is_static (&TYPE_FIELD (type, i)))
|
||
continue;
|
||
|
||
t = resolve_dynamic_type_internal (TYPE_FIELD_TYPE (resolved_type, i),
|
||
addr_stack, 0);
|
||
TYPE_FIELD_TYPE (resolved_type, i) = t;
|
||
if (TYPE_LENGTH (t) > max_len)
|
||
max_len = TYPE_LENGTH (t);
|
||
}
|
||
|
||
TYPE_LENGTH (resolved_type) = max_len;
|
||
return resolved_type;
|
||
}
|
||
|
||
/* Resolve dynamic bounds of members of the struct TYPE to static
|
||
bounds. ADDR_STACK is a stack of struct property_addr_info to
|
||
be used if needed during the dynamic resolution. */
|
||
|
||
static struct type *
|
||
resolve_dynamic_struct (struct type *type,
|
||
struct property_addr_info *addr_stack)
|
||
{
|
||
struct type *resolved_type;
|
||
int i;
|
||
unsigned resolved_type_bit_length = 0;
|
||
|
||
gdb_assert (TYPE_CODE (type) == TYPE_CODE_STRUCT);
|
||
gdb_assert (TYPE_NFIELDS (type) > 0);
|
||
|
||
resolved_type = copy_type (type);
|
||
TYPE_FIELDS (resolved_type)
|
||
= (struct field *) TYPE_ALLOC (resolved_type,
|
||
TYPE_NFIELDS (resolved_type)
|
||
* sizeof (struct field));
|
||
memcpy (TYPE_FIELDS (resolved_type),
|
||
TYPE_FIELDS (type),
|
||
TYPE_NFIELDS (resolved_type) * sizeof (struct field));
|
||
for (i = 0; i < TYPE_NFIELDS (resolved_type); ++i)
|
||
{
|
||
unsigned new_bit_length;
|
||
struct property_addr_info pinfo;
|
||
|
||
if (field_is_static (&TYPE_FIELD (type, i)))
|
||
continue;
|
||
|
||
/* As we know this field is not a static field, the field's
|
||
field_loc_kind should be FIELD_LOC_KIND_BITPOS. Verify
|
||
this is the case, but only trigger a simple error rather
|
||
than an internal error if that fails. While failing
|
||
that verification indicates a bug in our code, the error
|
||
is not severe enough to suggest to the user he stops
|
||
his debugging session because of it. */
|
||
if (TYPE_FIELD_LOC_KIND (type, i) != FIELD_LOC_KIND_BITPOS)
|
||
error (_("Cannot determine struct field location"
|
||
" (invalid location kind)"));
|
||
|
||
pinfo.type = check_typedef (TYPE_FIELD_TYPE (type, i));
|
||
pinfo.valaddr = addr_stack->valaddr;
|
||
pinfo.addr = addr_stack->addr;
|
||
pinfo.next = addr_stack;
|
||
|
||
TYPE_FIELD_TYPE (resolved_type, i)
|
||
= resolve_dynamic_type_internal (TYPE_FIELD_TYPE (resolved_type, i),
|
||
&pinfo, 0);
|
||
gdb_assert (TYPE_FIELD_LOC_KIND (resolved_type, i)
|
||
== FIELD_LOC_KIND_BITPOS);
|
||
|
||
new_bit_length = TYPE_FIELD_BITPOS (resolved_type, i);
|
||
if (TYPE_FIELD_BITSIZE (resolved_type, i) != 0)
|
||
new_bit_length += TYPE_FIELD_BITSIZE (resolved_type, i);
|
||
else
|
||
new_bit_length += (TYPE_LENGTH (TYPE_FIELD_TYPE (resolved_type, i))
|
||
* TARGET_CHAR_BIT);
|
||
|
||
/* Normally, we would use the position and size of the last field
|
||
to determine the size of the enclosing structure. But GCC seems
|
||
to be encoding the position of some fields incorrectly when
|
||
the struct contains a dynamic field that is not placed last.
|
||
So we compute the struct size based on the field that has
|
||
the highest position + size - probably the best we can do. */
|
||
if (new_bit_length > resolved_type_bit_length)
|
||
resolved_type_bit_length = new_bit_length;
|
||
}
|
||
|
||
TYPE_LENGTH (resolved_type)
|
||
= (resolved_type_bit_length + TARGET_CHAR_BIT - 1) / TARGET_CHAR_BIT;
|
||
|
||
/* The Ada language uses this field as a cache for static fixed types: reset
|
||
it as RESOLVED_TYPE must have its own static fixed type. */
|
||
TYPE_TARGET_TYPE (resolved_type) = NULL;
|
||
|
||
return resolved_type;
|
||
}
|
||
|
||
/* Worker for resolved_dynamic_type. */
|
||
|
||
static struct type *
|
||
resolve_dynamic_type_internal (struct type *type,
|
||
struct property_addr_info *addr_stack,
|
||
int top_level)
|
||
{
|
||
struct type *real_type = check_typedef (type);
|
||
struct type *resolved_type = type;
|
||
struct dynamic_prop *prop;
|
||
CORE_ADDR value;
|
||
|
||
if (!is_dynamic_type_internal (real_type, top_level))
|
||
return type;
|
||
|
||
if (TYPE_CODE (type) == TYPE_CODE_TYPEDEF)
|
||
{
|
||
resolved_type = copy_type (type);
|
||
TYPE_TARGET_TYPE (resolved_type)
|
||
= resolve_dynamic_type_internal (TYPE_TARGET_TYPE (type), addr_stack,
|
||
top_level);
|
||
}
|
||
else
|
||
{
|
||
/* Before trying to resolve TYPE, make sure it is not a stub. */
|
||
type = real_type;
|
||
|
||
switch (TYPE_CODE (type))
|
||
{
|
||
case TYPE_CODE_REF:
|
||
{
|
||
struct property_addr_info pinfo;
|
||
|
||
pinfo.type = check_typedef (TYPE_TARGET_TYPE (type));
|
||
pinfo.valaddr = NULL;
|
||
if (addr_stack->valaddr != NULL)
|
||
pinfo.addr = extract_typed_address (addr_stack->valaddr, type);
|
||
else
|
||
pinfo.addr = read_memory_typed_address (addr_stack->addr, type);
|
||
pinfo.next = addr_stack;
|
||
|
||
resolved_type = copy_type (type);
|
||
TYPE_TARGET_TYPE (resolved_type)
|
||
= resolve_dynamic_type_internal (TYPE_TARGET_TYPE (type),
|
||
&pinfo, top_level);
|
||
break;
|
||
}
|
||
|
||
case TYPE_CODE_ARRAY:
|
||
resolved_type = resolve_dynamic_array (type, addr_stack);
|
||
break;
|
||
|
||
case TYPE_CODE_RANGE:
|
||
resolved_type = resolve_dynamic_range (type, addr_stack);
|
||
break;
|
||
|
||
case TYPE_CODE_UNION:
|
||
resolved_type = resolve_dynamic_union (type, addr_stack);
|
||
break;
|
||
|
||
case TYPE_CODE_STRUCT:
|
||
resolved_type = resolve_dynamic_struct (type, addr_stack);
|
||
break;
|
||
}
|
||
}
|
||
|
||
/* Resolve data_location attribute. */
|
||
prop = TYPE_DATA_LOCATION (resolved_type);
|
||
if (prop != NULL
|
||
&& dwarf2_evaluate_property (prop, NULL, addr_stack, &value))
|
||
{
|
||
TYPE_DYN_PROP_ADDR (prop) = value;
|
||
TYPE_DYN_PROP_KIND (prop) = PROP_CONST;
|
||
}
|
||
|
||
return resolved_type;
|
||
}
|
||
|
||
/* See gdbtypes.h */
|
||
|
||
struct type *
|
||
resolve_dynamic_type (struct type *type, const gdb_byte *valaddr,
|
||
CORE_ADDR addr)
|
||
{
|
||
struct property_addr_info pinfo
|
||
= {check_typedef (type), valaddr, addr, NULL};
|
||
|
||
return resolve_dynamic_type_internal (type, &pinfo, 1);
|
||
}
|
||
|
||
/* See gdbtypes.h */
|
||
|
||
struct dynamic_prop *
|
||
get_dyn_prop (enum dynamic_prop_node_kind prop_kind, const struct type *type)
|
||
{
|
||
struct dynamic_prop_list *node = TYPE_DYN_PROP_LIST (type);
|
||
|
||
while (node != NULL)
|
||
{
|
||
if (node->prop_kind == prop_kind)
|
||
return &node->prop;
|
||
node = node->next;
|
||
}
|
||
return NULL;
|
||
}
|
||
|
||
/* See gdbtypes.h */
|
||
|
||
void
|
||
add_dyn_prop (enum dynamic_prop_node_kind prop_kind, struct dynamic_prop prop,
|
||
struct type *type, struct objfile *objfile)
|
||
{
|
||
struct dynamic_prop_list *temp;
|
||
|
||
gdb_assert (TYPE_OBJFILE_OWNED (type));
|
||
|
||
temp = XOBNEW (&objfile->objfile_obstack, struct dynamic_prop_list);
|
||
temp->prop_kind = prop_kind;
|
||
temp->prop = prop;
|
||
temp->next = TYPE_DYN_PROP_LIST (type);
|
||
|
||
TYPE_DYN_PROP_LIST (type) = temp;
|
||
}
|
||
|
||
|
||
/* Find the real type of TYPE. This function returns the real type,
|
||
after removing all layers of typedefs, and completing opaque or stub
|
||
types. Completion changes the TYPE argument, but stripping of
|
||
typedefs does not.
|
||
|
||
Instance flags (e.g. const/volatile) are preserved as typedefs are
|
||
stripped. If necessary a new qualified form of the underlying type
|
||
is created.
|
||
|
||
NOTE: This will return a typedef if TYPE_TARGET_TYPE for the typedef has
|
||
not been computed and we're either in the middle of reading symbols, or
|
||
there was no name for the typedef in the debug info.
|
||
|
||
NOTE: Lookup of opaque types can throw errors for invalid symbol files.
|
||
QUITs in the symbol reading code can also throw.
|
||
Thus this function can throw an exception.
|
||
|
||
If TYPE is a TYPE_CODE_TYPEDEF, its length is updated to the length of
|
||
the target type.
|
||
|
||
If this is a stubbed struct (i.e. declared as struct foo *), see if
|
||
we can find a full definition in some other file. If so, copy this
|
||
definition, so we can use it in future. There used to be a comment
|
||
(but not any code) that if we don't find a full definition, we'd
|
||
set a flag so we don't spend time in the future checking the same
|
||
type. That would be a mistake, though--we might load in more
|
||
symbols which contain a full definition for the type. */
|
||
|
||
struct type *
|
||
check_typedef (struct type *type)
|
||
{
|
||
struct type *orig_type = type;
|
||
/* While we're removing typedefs, we don't want to lose qualifiers.
|
||
E.g., const/volatile. */
|
||
int instance_flags = TYPE_INSTANCE_FLAGS (type);
|
||
|
||
gdb_assert (type);
|
||
|
||
while (TYPE_CODE (type) == TYPE_CODE_TYPEDEF)
|
||
{
|
||
if (!TYPE_TARGET_TYPE (type))
|
||
{
|
||
const char *name;
|
||
struct symbol *sym;
|
||
|
||
/* It is dangerous to call lookup_symbol if we are currently
|
||
reading a symtab. Infinite recursion is one danger. */
|
||
if (currently_reading_symtab)
|
||
return make_qualified_type (type, instance_flags, NULL);
|
||
|
||
name = type_name_no_tag (type);
|
||
/* FIXME: shouldn't we separately check the TYPE_NAME and
|
||
the TYPE_TAG_NAME, and look in STRUCT_DOMAIN and/or
|
||
VAR_DOMAIN as appropriate? (this code was written before
|
||
TYPE_NAME and TYPE_TAG_NAME were separate). */
|
||
if (name == NULL)
|
||
{
|
||
stub_noname_complaint ();
|
||
return make_qualified_type (type, instance_flags, NULL);
|
||
}
|
||
sym = lookup_symbol (name, 0, STRUCT_DOMAIN, 0).symbol;
|
||
if (sym)
|
||
TYPE_TARGET_TYPE (type) = SYMBOL_TYPE (sym);
|
||
else /* TYPE_CODE_UNDEF */
|
||
TYPE_TARGET_TYPE (type) = alloc_type_arch (get_type_arch (type));
|
||
}
|
||
type = TYPE_TARGET_TYPE (type);
|
||
|
||
/* Preserve the instance flags as we traverse down the typedef chain.
|
||
|
||
Handling address spaces/classes is nasty, what do we do if there's a
|
||
conflict?
|
||
E.g., what if an outer typedef marks the type as class_1 and an inner
|
||
typedef marks the type as class_2?
|
||
This is the wrong place to do such error checking. We leave it to
|
||
the code that created the typedef in the first place to flag the
|
||
error. We just pick the outer address space (akin to letting the
|
||
outer cast in a chain of casting win), instead of assuming
|
||
"it can't happen". */
|
||
{
|
||
const int ALL_SPACES = (TYPE_INSTANCE_FLAG_CODE_SPACE
|
||
| TYPE_INSTANCE_FLAG_DATA_SPACE);
|
||
const int ALL_CLASSES = TYPE_INSTANCE_FLAG_ADDRESS_CLASS_ALL;
|
||
int new_instance_flags = TYPE_INSTANCE_FLAGS (type);
|
||
|
||
/* Treat code vs data spaces and address classes separately. */
|
||
if ((instance_flags & ALL_SPACES) != 0)
|
||
new_instance_flags &= ~ALL_SPACES;
|
||
if ((instance_flags & ALL_CLASSES) != 0)
|
||
new_instance_flags &= ~ALL_CLASSES;
|
||
|
||
instance_flags |= new_instance_flags;
|
||
}
|
||
}
|
||
|
||
/* If this is a struct/class/union with no fields, then check
|
||
whether a full definition exists somewhere else. This is for
|
||
systems where a type definition with no fields is issued for such
|
||
types, instead of identifying them as stub types in the first
|
||
place. */
|
||
|
||
if (TYPE_IS_OPAQUE (type)
|
||
&& opaque_type_resolution
|
||
&& !currently_reading_symtab)
|
||
{
|
||
const char *name = type_name_no_tag (type);
|
||
struct type *newtype;
|
||
|
||
if (name == NULL)
|
||
{
|
||
stub_noname_complaint ();
|
||
return make_qualified_type (type, instance_flags, NULL);
|
||
}
|
||
newtype = lookup_transparent_type (name);
|
||
|
||
if (newtype)
|
||
{
|
||
/* If the resolved type and the stub are in the same
|
||
objfile, then replace the stub type with the real deal.
|
||
But if they're in separate objfiles, leave the stub
|
||
alone; we'll just look up the transparent type every time
|
||
we call check_typedef. We can't create pointers between
|
||
types allocated to different objfiles, since they may
|
||
have different lifetimes. Trying to copy NEWTYPE over to
|
||
TYPE's objfile is pointless, too, since you'll have to
|
||
move over any other types NEWTYPE refers to, which could
|
||
be an unbounded amount of stuff. */
|
||
if (TYPE_OBJFILE (newtype) == TYPE_OBJFILE (type))
|
||
type = make_qualified_type (newtype,
|
||
TYPE_INSTANCE_FLAGS (type),
|
||
type);
|
||
else
|
||
type = newtype;
|
||
}
|
||
}
|
||
/* Otherwise, rely on the stub flag being set for opaque/stubbed
|
||
types. */
|
||
else if (TYPE_STUB (type) && !currently_reading_symtab)
|
||
{
|
||
const char *name = type_name_no_tag (type);
|
||
/* FIXME: shouldn't we separately check the TYPE_NAME and the
|
||
TYPE_TAG_NAME, and look in STRUCT_DOMAIN and/or VAR_DOMAIN
|
||
as appropriate? (this code was written before TYPE_NAME and
|
||
TYPE_TAG_NAME were separate). */
|
||
struct symbol *sym;
|
||
|
||
if (name == NULL)
|
||
{
|
||
stub_noname_complaint ();
|
||
return make_qualified_type (type, instance_flags, NULL);
|
||
}
|
||
sym = lookup_symbol (name, 0, STRUCT_DOMAIN, 0).symbol;
|
||
if (sym)
|
||
{
|
||
/* Same as above for opaque types, we can replace the stub
|
||
with the complete type only if they are in the same
|
||
objfile. */
|
||
if (TYPE_OBJFILE (SYMBOL_TYPE(sym)) == TYPE_OBJFILE (type))
|
||
type = make_qualified_type (SYMBOL_TYPE (sym),
|
||
TYPE_INSTANCE_FLAGS (type),
|
||
type);
|
||
else
|
||
type = SYMBOL_TYPE (sym);
|
||
}
|
||
}
|
||
|
||
if (TYPE_TARGET_STUB (type))
|
||
{
|
||
struct type *range_type;
|
||
struct type *target_type = check_typedef (TYPE_TARGET_TYPE (type));
|
||
|
||
if (TYPE_STUB (target_type) || TYPE_TARGET_STUB (target_type))
|
||
{
|
||
/* Nothing we can do. */
|
||
}
|
||
else if (TYPE_CODE (type) == TYPE_CODE_RANGE)
|
||
{
|
||
TYPE_LENGTH (type) = TYPE_LENGTH (target_type);
|
||
TYPE_TARGET_STUB (type) = 0;
|
||
}
|
||
}
|
||
|
||
type = make_qualified_type (type, instance_flags, NULL);
|
||
|
||
/* Cache TYPE_LENGTH for future use. */
|
||
TYPE_LENGTH (orig_type) = TYPE_LENGTH (type);
|
||
|
||
return type;
|
||
}
|
||
|
||
/* Parse a type expression in the string [P..P+LENGTH). If an error
|
||
occurs, silently return a void type. */
|
||
|
||
static struct type *
|
||
safe_parse_type (struct gdbarch *gdbarch, char *p, int length)
|
||
{
|
||
struct ui_file *saved_gdb_stderr;
|
||
struct type *type = NULL; /* Initialize to keep gcc happy. */
|
||
|
||
/* Suppress error messages. */
|
||
saved_gdb_stderr = gdb_stderr;
|
||
gdb_stderr = ui_file_new ();
|
||
|
||
/* Call parse_and_eval_type() without fear of longjmp()s. */
|
||
TRY
|
||
{
|
||
type = parse_and_eval_type (p, length);
|
||
}
|
||
CATCH (except, RETURN_MASK_ERROR)
|
||
{
|
||
type = builtin_type (gdbarch)->builtin_void;
|
||
}
|
||
END_CATCH
|
||
|
||
/* Stop suppressing error messages. */
|
||
ui_file_delete (gdb_stderr);
|
||
gdb_stderr = saved_gdb_stderr;
|
||
|
||
return type;
|
||
}
|
||
|
||
/* Ugly hack to convert method stubs into method types.
|
||
|
||
He ain't kiddin'. This demangles the name of the method into a
|
||
string including argument types, parses out each argument type,
|
||
generates a string casting a zero to that type, evaluates the
|
||
string, and stuffs the resulting type into an argtype vector!!!
|
||
Then it knows the type of the whole function (including argument
|
||
types for overloading), which info used to be in the stab's but was
|
||
removed to hack back the space required for them. */
|
||
|
||
static void
|
||
check_stub_method (struct type *type, int method_id, int signature_id)
|
||
{
|
||
struct gdbarch *gdbarch = get_type_arch (type);
|
||
struct fn_field *f;
|
||
char *mangled_name = gdb_mangle_name (type, method_id, signature_id);
|
||
char *demangled_name = gdb_demangle (mangled_name,
|
||
DMGL_PARAMS | DMGL_ANSI);
|
||
char *argtypetext, *p;
|
||
int depth = 0, argcount = 1;
|
||
struct field *argtypes;
|
||
struct type *mtype;
|
||
|
||
/* Make sure we got back a function string that we can use. */
|
||
if (demangled_name)
|
||
p = strchr (demangled_name, '(');
|
||
else
|
||
p = NULL;
|
||
|
||
if (demangled_name == NULL || p == NULL)
|
||
error (_("Internal: Cannot demangle mangled name `%s'."),
|
||
mangled_name);
|
||
|
||
/* Now, read in the parameters that define this type. */
|
||
p += 1;
|
||
argtypetext = p;
|
||
while (*p)
|
||
{
|
||
if (*p == '(' || *p == '<')
|
||
{
|
||
depth += 1;
|
||
}
|
||
else if (*p == ')' || *p == '>')
|
||
{
|
||
depth -= 1;
|
||
}
|
||
else if (*p == ',' && depth == 0)
|
||
{
|
||
argcount += 1;
|
||
}
|
||
|
||
p += 1;
|
||
}
|
||
|
||
/* If we read one argument and it was ``void'', don't count it. */
|
||
if (startswith (argtypetext, "(void)"))
|
||
argcount -= 1;
|
||
|
||
/* We need one extra slot, for the THIS pointer. */
|
||
|
||
argtypes = (struct field *)
|
||
TYPE_ALLOC (type, (argcount + 1) * sizeof (struct field));
|
||
p = argtypetext;
|
||
|
||
/* Add THIS pointer for non-static methods. */
|
||
f = TYPE_FN_FIELDLIST1 (type, method_id);
|
||
if (TYPE_FN_FIELD_STATIC_P (f, signature_id))
|
||
argcount = 0;
|
||
else
|
||
{
|
||
argtypes[0].type = lookup_pointer_type (type);
|
||
argcount = 1;
|
||
}
|
||
|
||
if (*p != ')') /* () means no args, skip while. */
|
||
{
|
||
depth = 0;
|
||
while (*p)
|
||
{
|
||
if (depth <= 0 && (*p == ',' || *p == ')'))
|
||
{
|
||
/* Avoid parsing of ellipsis, they will be handled below.
|
||
Also avoid ``void'' as above. */
|
||
if (strncmp (argtypetext, "...", p - argtypetext) != 0
|
||
&& strncmp (argtypetext, "void", p - argtypetext) != 0)
|
||
{
|
||
argtypes[argcount].type =
|
||
safe_parse_type (gdbarch, argtypetext, p - argtypetext);
|
||
argcount += 1;
|
||
}
|
||
argtypetext = p + 1;
|
||
}
|
||
|
||
if (*p == '(' || *p == '<')
|
||
{
|
||
depth += 1;
|
||
}
|
||
else if (*p == ')' || *p == '>')
|
||
{
|
||
depth -= 1;
|
||
}
|
||
|
||
p += 1;
|
||
}
|
||
}
|
||
|
||
TYPE_FN_FIELD_PHYSNAME (f, signature_id) = mangled_name;
|
||
|
||
/* Now update the old "stub" type into a real type. */
|
||
mtype = TYPE_FN_FIELD_TYPE (f, signature_id);
|
||
/* MTYPE may currently be a function (TYPE_CODE_FUNC).
|
||
We want a method (TYPE_CODE_METHOD). */
|
||
smash_to_method_type (mtype, type, TYPE_TARGET_TYPE (mtype),
|
||
argtypes, argcount, p[-2] == '.');
|
||
TYPE_STUB (mtype) = 0;
|
||
TYPE_FN_FIELD_STUB (f, signature_id) = 0;
|
||
|
||
xfree (demangled_name);
|
||
}
|
||
|
||
/* This is the external interface to check_stub_method, above. This
|
||
function unstubs all of the signatures for TYPE's METHOD_ID method
|
||
name. After calling this function TYPE_FN_FIELD_STUB will be
|
||
cleared for each signature and TYPE_FN_FIELDLIST_NAME will be
|
||
correct.
|
||
|
||
This function unfortunately can not die until stabs do. */
|
||
|
||
void
|
||
check_stub_method_group (struct type *type, int method_id)
|
||
{
|
||
int len = TYPE_FN_FIELDLIST_LENGTH (type, method_id);
|
||
struct fn_field *f = TYPE_FN_FIELDLIST1 (type, method_id);
|
||
int j, found_stub = 0;
|
||
|
||
for (j = 0; j < len; j++)
|
||
if (TYPE_FN_FIELD_STUB (f, j))
|
||
{
|
||
found_stub = 1;
|
||
check_stub_method (type, method_id, j);
|
||
}
|
||
|
||
/* GNU v3 methods with incorrect names were corrected when we read
|
||
in type information, because it was cheaper to do it then. The
|
||
only GNU v2 methods with incorrect method names are operators and
|
||
destructors; destructors were also corrected when we read in type
|
||
information.
|
||
|
||
Therefore the only thing we need to handle here are v2 operator
|
||
names. */
|
||
if (found_stub && !startswith (TYPE_FN_FIELD_PHYSNAME (f, 0), "_Z"))
|
||
{
|
||
int ret;
|
||
char dem_opname[256];
|
||
|
||
ret = cplus_demangle_opname (TYPE_FN_FIELDLIST_NAME (type,
|
||
method_id),
|
||
dem_opname, DMGL_ANSI);
|
||
if (!ret)
|
||
ret = cplus_demangle_opname (TYPE_FN_FIELDLIST_NAME (type,
|
||
method_id),
|
||
dem_opname, 0);
|
||
if (ret)
|
||
TYPE_FN_FIELDLIST_NAME (type, method_id) = xstrdup (dem_opname);
|
||
}
|
||
}
|
||
|
||
/* Ensure it is in .rodata (if available) by workarounding GCC PR 44690. */
|
||
const struct cplus_struct_type cplus_struct_default = { };
|
||
|
||
void
|
||
allocate_cplus_struct_type (struct type *type)
|
||
{
|
||
if (HAVE_CPLUS_STRUCT (type))
|
||
/* Structure was already allocated. Nothing more to do. */
|
||
return;
|
||
|
||
TYPE_SPECIFIC_FIELD (type) = TYPE_SPECIFIC_CPLUS_STUFF;
|
||
TYPE_RAW_CPLUS_SPECIFIC (type) = (struct cplus_struct_type *)
|
||
TYPE_ALLOC (type, sizeof (struct cplus_struct_type));
|
||
*(TYPE_RAW_CPLUS_SPECIFIC (type)) = cplus_struct_default;
|
||
set_type_vptr_fieldno (type, -1);
|
||
}
|
||
|
||
const struct gnat_aux_type gnat_aux_default =
|
||
{ NULL };
|
||
|
||
/* Set the TYPE's type-specific kind to TYPE_SPECIFIC_GNAT_STUFF,
|
||
and allocate the associated gnat-specific data. The gnat-specific
|
||
data is also initialized to gnat_aux_default. */
|
||
|
||
void
|
||
allocate_gnat_aux_type (struct type *type)
|
||
{
|
||
TYPE_SPECIFIC_FIELD (type) = TYPE_SPECIFIC_GNAT_STUFF;
|
||
TYPE_GNAT_SPECIFIC (type) = (struct gnat_aux_type *)
|
||
TYPE_ALLOC (type, sizeof (struct gnat_aux_type));
|
||
*(TYPE_GNAT_SPECIFIC (type)) = gnat_aux_default;
|
||
}
|
||
|
||
/* Helper function to initialize the standard scalar types.
|
||
|
||
If NAME is non-NULL, then it is used to initialize the type name.
|
||
Note that NAME is not copied; it is required to have a lifetime at
|
||
least as long as OBJFILE. */
|
||
|
||
struct type *
|
||
init_type (enum type_code code, int length, int flags,
|
||
const char *name, struct objfile *objfile)
|
||
{
|
||
struct type *type;
|
||
|
||
type = alloc_type (objfile);
|
||
TYPE_CODE (type) = code;
|
||
TYPE_LENGTH (type) = length;
|
||
|
||
gdb_assert (!(flags & (TYPE_FLAG_MIN - 1)));
|
||
if (flags & TYPE_FLAG_UNSIGNED)
|
||
TYPE_UNSIGNED (type) = 1;
|
||
if (flags & TYPE_FLAG_NOSIGN)
|
||
TYPE_NOSIGN (type) = 1;
|
||
if (flags & TYPE_FLAG_STUB)
|
||
TYPE_STUB (type) = 1;
|
||
if (flags & TYPE_FLAG_TARGET_STUB)
|
||
TYPE_TARGET_STUB (type) = 1;
|
||
if (flags & TYPE_FLAG_STATIC)
|
||
TYPE_STATIC (type) = 1;
|
||
if (flags & TYPE_FLAG_PROTOTYPED)
|
||
TYPE_PROTOTYPED (type) = 1;
|
||
if (flags & TYPE_FLAG_INCOMPLETE)
|
||
TYPE_INCOMPLETE (type) = 1;
|
||
if (flags & TYPE_FLAG_VARARGS)
|
||
TYPE_VARARGS (type) = 1;
|
||
if (flags & TYPE_FLAG_VECTOR)
|
||
TYPE_VECTOR (type) = 1;
|
||
if (flags & TYPE_FLAG_STUB_SUPPORTED)
|
||
TYPE_STUB_SUPPORTED (type) = 1;
|
||
if (flags & TYPE_FLAG_FIXED_INSTANCE)
|
||
TYPE_FIXED_INSTANCE (type) = 1;
|
||
if (flags & TYPE_FLAG_GNU_IFUNC)
|
||
TYPE_GNU_IFUNC (type) = 1;
|
||
|
||
TYPE_NAME (type) = name;
|
||
|
||
/* C++ fancies. */
|
||
|
||
if (name && strcmp (name, "char") == 0)
|
||
TYPE_NOSIGN (type) = 1;
|
||
|
||
switch (code)
|
||
{
|
||
case TYPE_CODE_STRUCT:
|
||
case TYPE_CODE_UNION:
|
||
case TYPE_CODE_NAMESPACE:
|
||
INIT_CPLUS_SPECIFIC (type);
|
||
break;
|
||
case TYPE_CODE_FLT:
|
||
TYPE_SPECIFIC_FIELD (type) = TYPE_SPECIFIC_FLOATFORMAT;
|
||
break;
|
||
case TYPE_CODE_FUNC:
|
||
INIT_FUNC_SPECIFIC (type);
|
||
break;
|
||
}
|
||
return type;
|
||
}
|
||
|
||
/* Queries on types. */
|
||
|
||
int
|
||
can_dereference (struct type *t)
|
||
{
|
||
/* FIXME: Should we return true for references as well as
|
||
pointers? */
|
||
t = check_typedef (t);
|
||
return
|
||
(t != NULL
|
||
&& TYPE_CODE (t) == TYPE_CODE_PTR
|
||
&& TYPE_CODE (TYPE_TARGET_TYPE (t)) != TYPE_CODE_VOID);
|
||
}
|
||
|
||
int
|
||
is_integral_type (struct type *t)
|
||
{
|
||
t = check_typedef (t);
|
||
return
|
||
((t != NULL)
|
||
&& ((TYPE_CODE (t) == TYPE_CODE_INT)
|
||
|| (TYPE_CODE (t) == TYPE_CODE_ENUM)
|
||
|| (TYPE_CODE (t) == TYPE_CODE_FLAGS)
|
||
|| (TYPE_CODE (t) == TYPE_CODE_CHAR)
|
||
|| (TYPE_CODE (t) == TYPE_CODE_RANGE)
|
||
|| (TYPE_CODE (t) == TYPE_CODE_BOOL)));
|
||
}
|
||
|
||
/* Return true if TYPE is scalar. */
|
||
|
||
static int
|
||
is_scalar_type (struct type *type)
|
||
{
|
||
type = check_typedef (type);
|
||
|
||
switch (TYPE_CODE (type))
|
||
{
|
||
case TYPE_CODE_ARRAY:
|
||
case TYPE_CODE_STRUCT:
|
||
case TYPE_CODE_UNION:
|
||
case TYPE_CODE_SET:
|
||
case TYPE_CODE_STRING:
|
||
return 0;
|
||
default:
|
||
return 1;
|
||
}
|
||
}
|
||
|
||
/* Return true if T is scalar, or a composite type which in practice has
|
||
the memory layout of a scalar type. E.g., an array or struct with only
|
||
one scalar element inside it, or a union with only scalar elements. */
|
||
|
||
int
|
||
is_scalar_type_recursive (struct type *t)
|
||
{
|
||
t = check_typedef (t);
|
||
|
||
if (is_scalar_type (t))
|
||
return 1;
|
||
/* Are we dealing with an array or string of known dimensions? */
|
||
else if ((TYPE_CODE (t) == TYPE_CODE_ARRAY
|
||
|| TYPE_CODE (t) == TYPE_CODE_STRING) && TYPE_NFIELDS (t) == 1
|
||
&& TYPE_CODE (TYPE_INDEX_TYPE (t)) == TYPE_CODE_RANGE)
|
||
{
|
||
LONGEST low_bound, high_bound;
|
||
struct type *elt_type = check_typedef (TYPE_TARGET_TYPE (t));
|
||
|
||
get_discrete_bounds (TYPE_INDEX_TYPE (t), &low_bound, &high_bound);
|
||
|
||
return high_bound == low_bound && is_scalar_type_recursive (elt_type);
|
||
}
|
||
/* Are we dealing with a struct with one element? */
|
||
else if (TYPE_CODE (t) == TYPE_CODE_STRUCT && TYPE_NFIELDS (t) == 1)
|
||
return is_scalar_type_recursive (TYPE_FIELD_TYPE (t, 0));
|
||
else if (TYPE_CODE (t) == TYPE_CODE_UNION)
|
||
{
|
||
int i, n = TYPE_NFIELDS (t);
|
||
|
||
/* If all elements of the union are scalar, then the union is scalar. */
|
||
for (i = 0; i < n; i++)
|
||
if (!is_scalar_type_recursive (TYPE_FIELD_TYPE (t, i)))
|
||
return 0;
|
||
|
||
return 1;
|
||
}
|
||
|
||
return 0;
|
||
}
|
||
|
||
/* Return true is T is a class or a union. False otherwise. */
|
||
|
||
int
|
||
class_or_union_p (const struct type *t)
|
||
{
|
||
return (TYPE_CODE (t) == TYPE_CODE_STRUCT
|
||
|| TYPE_CODE (t) == TYPE_CODE_UNION);
|
||
}
|
||
|
||
/* A helper function which returns true if types A and B represent the
|
||
"same" class type. This is true if the types have the same main
|
||
type, or the same name. */
|
||
|
||
int
|
||
class_types_same_p (const struct type *a, const struct type *b)
|
||
{
|
||
return (TYPE_MAIN_TYPE (a) == TYPE_MAIN_TYPE (b)
|
||
|| (TYPE_NAME (a) && TYPE_NAME (b)
|
||
&& !strcmp (TYPE_NAME (a), TYPE_NAME (b))));
|
||
}
|
||
|
||
/* If BASE is an ancestor of DCLASS return the distance between them.
|
||
otherwise return -1;
|
||
eg:
|
||
|
||
class A {};
|
||
class B: public A {};
|
||
class C: public B {};
|
||
class D: C {};
|
||
|
||
distance_to_ancestor (A, A, 0) = 0
|
||
distance_to_ancestor (A, B, 0) = 1
|
||
distance_to_ancestor (A, C, 0) = 2
|
||
distance_to_ancestor (A, D, 0) = 3
|
||
|
||
If PUBLIC is 1 then only public ancestors are considered,
|
||
and the function returns the distance only if BASE is a public ancestor
|
||
of DCLASS.
|
||
Eg:
|
||
|
||
distance_to_ancestor (A, D, 1) = -1. */
|
||
|
||
static int
|
||
distance_to_ancestor (struct type *base, struct type *dclass, int is_public)
|
||
{
|
||
int i;
|
||
int d;
|
||
|
||
base = check_typedef (base);
|
||
dclass = check_typedef (dclass);
|
||
|
||
if (class_types_same_p (base, dclass))
|
||
return 0;
|
||
|
||
for (i = 0; i < TYPE_N_BASECLASSES (dclass); i++)
|
||
{
|
||
if (is_public && ! BASETYPE_VIA_PUBLIC (dclass, i))
|
||
continue;
|
||
|
||
d = distance_to_ancestor (base, TYPE_BASECLASS (dclass, i), is_public);
|
||
if (d >= 0)
|
||
return 1 + d;
|
||
}
|
||
|
||
return -1;
|
||
}
|
||
|
||
/* Check whether BASE is an ancestor or base class or DCLASS
|
||
Return 1 if so, and 0 if not.
|
||
Note: If BASE and DCLASS are of the same type, this function
|
||
will return 1. So for some class A, is_ancestor (A, A) will
|
||
return 1. */
|
||
|
||
int
|
||
is_ancestor (struct type *base, struct type *dclass)
|
||
{
|
||
return distance_to_ancestor (base, dclass, 0) >= 0;
|
||
}
|
||
|
||
/* Like is_ancestor, but only returns true when BASE is a public
|
||
ancestor of DCLASS. */
|
||
|
||
int
|
||
is_public_ancestor (struct type *base, struct type *dclass)
|
||
{
|
||
return distance_to_ancestor (base, dclass, 1) >= 0;
|
||
}
|
||
|
||
/* A helper function for is_unique_ancestor. */
|
||
|
||
static int
|
||
is_unique_ancestor_worker (struct type *base, struct type *dclass,
|
||
int *offset,
|
||
const gdb_byte *valaddr, int embedded_offset,
|
||
CORE_ADDR address, struct value *val)
|
||
{
|
||
int i, count = 0;
|
||
|
||
base = check_typedef (base);
|
||
dclass = check_typedef (dclass);
|
||
|
||
for (i = 0; i < TYPE_N_BASECLASSES (dclass) && count < 2; ++i)
|
||
{
|
||
struct type *iter;
|
||
int this_offset;
|
||
|
||
iter = check_typedef (TYPE_BASECLASS (dclass, i));
|
||
|
||
this_offset = baseclass_offset (dclass, i, valaddr, embedded_offset,
|
||
address, val);
|
||
|
||
if (class_types_same_p (base, iter))
|
||
{
|
||
/* If this is the first subclass, set *OFFSET and set count
|
||
to 1. Otherwise, if this is at the same offset as
|
||
previous instances, do nothing. Otherwise, increment
|
||
count. */
|
||
if (*offset == -1)
|
||
{
|
||
*offset = this_offset;
|
||
count = 1;
|
||
}
|
||
else if (this_offset == *offset)
|
||
{
|
||
/* Nothing. */
|
||
}
|
||
else
|
||
++count;
|
||
}
|
||
else
|
||
count += is_unique_ancestor_worker (base, iter, offset,
|
||
valaddr,
|
||
embedded_offset + this_offset,
|
||
address, val);
|
||
}
|
||
|
||
return count;
|
||
}
|
||
|
||
/* Like is_ancestor, but only returns true if BASE is a unique base
|
||
class of the type of VAL. */
|
||
|
||
int
|
||
is_unique_ancestor (struct type *base, struct value *val)
|
||
{
|
||
int offset = -1;
|
||
|
||
return is_unique_ancestor_worker (base, value_type (val), &offset,
|
||
value_contents_for_printing (val),
|
||
value_embedded_offset (val),
|
||
value_address (val), val) == 1;
|
||
}
|
||
|
||
|
||
/* Overload resolution. */
|
||
|
||
/* Return the sum of the rank of A with the rank of B. */
|
||
|
||
struct rank
|
||
sum_ranks (struct rank a, struct rank b)
|
||
{
|
||
struct rank c;
|
||
c.rank = a.rank + b.rank;
|
||
c.subrank = a.subrank + b.subrank;
|
||
return c;
|
||
}
|
||
|
||
/* Compare rank A and B and return:
|
||
0 if a = b
|
||
1 if a is better than b
|
||
-1 if b is better than a. */
|
||
|
||
int
|
||
compare_ranks (struct rank a, struct rank b)
|
||
{
|
||
if (a.rank == b.rank)
|
||
{
|
||
if (a.subrank == b.subrank)
|
||
return 0;
|
||
if (a.subrank < b.subrank)
|
||
return 1;
|
||
if (a.subrank > b.subrank)
|
||
return -1;
|
||
}
|
||
|
||
if (a.rank < b.rank)
|
||
return 1;
|
||
|
||
/* a.rank > b.rank */
|
||
return -1;
|
||
}
|
||
|
||
/* Functions for overload resolution begin here. */
|
||
|
||
/* Compare two badness vectors A and B and return the result.
|
||
0 => A and B are identical
|
||
1 => A and B are incomparable
|
||
2 => A is better than B
|
||
3 => A is worse than B */
|
||
|
||
int
|
||
compare_badness (struct badness_vector *a, struct badness_vector *b)
|
||
{
|
||
int i;
|
||
int tmp;
|
||
short found_pos = 0; /* any positives in c? */
|
||
short found_neg = 0; /* any negatives in c? */
|
||
|
||
/* differing lengths => incomparable */
|
||
if (a->length != b->length)
|
||
return 1;
|
||
|
||
/* Subtract b from a */
|
||
for (i = 0; i < a->length; i++)
|
||
{
|
||
tmp = compare_ranks (b->rank[i], a->rank[i]);
|
||
if (tmp > 0)
|
||
found_pos = 1;
|
||
else if (tmp < 0)
|
||
found_neg = 1;
|
||
}
|
||
|
||
if (found_pos)
|
||
{
|
||
if (found_neg)
|
||
return 1; /* incomparable */
|
||
else
|
||
return 3; /* A > B */
|
||
}
|
||
else
|
||
/* no positives */
|
||
{
|
||
if (found_neg)
|
||
return 2; /* A < B */
|
||
else
|
||
return 0; /* A == B */
|
||
}
|
||
}
|
||
|
||
/* Rank a function by comparing its parameter types (PARMS, length
|
||
NPARMS), to the types of an argument list (ARGS, length NARGS).
|
||
Return a pointer to a badness vector. This has NARGS + 1
|
||
entries. */
|
||
|
||
struct badness_vector *
|
||
rank_function (struct type **parms, int nparms,
|
||
struct value **args, int nargs)
|
||
{
|
||
int i;
|
||
struct badness_vector *bv = XNEW (struct badness_vector);
|
||
int min_len = nparms < nargs ? nparms : nargs;
|
||
|
||
bv->length = nargs + 1; /* add 1 for the length-match rank. */
|
||
bv->rank = XNEWVEC (struct rank, nargs + 1);
|
||
|
||
/* First compare the lengths of the supplied lists.
|
||
If there is a mismatch, set it to a high value. */
|
||
|
||
/* pai/1997-06-03 FIXME: when we have debug info about default
|
||
arguments and ellipsis parameter lists, we should consider those
|
||
and rank the length-match more finely. */
|
||
|
||
LENGTH_MATCH (bv) = (nargs != nparms)
|
||
? LENGTH_MISMATCH_BADNESS
|
||
: EXACT_MATCH_BADNESS;
|
||
|
||
/* Now rank all the parameters of the candidate function. */
|
||
for (i = 1; i <= min_len; i++)
|
||
bv->rank[i] = rank_one_type (parms[i - 1], value_type (args[i - 1]),
|
||
args[i - 1]);
|
||
|
||
/* If more arguments than parameters, add dummy entries. */
|
||
for (i = min_len + 1; i <= nargs; i++)
|
||
bv->rank[i] = TOO_FEW_PARAMS_BADNESS;
|
||
|
||
return bv;
|
||
}
|
||
|
||
/* Compare the names of two integer types, assuming that any sign
|
||
qualifiers have been checked already. We do it this way because
|
||
there may be an "int" in the name of one of the types. */
|
||
|
||
static int
|
||
integer_types_same_name_p (const char *first, const char *second)
|
||
{
|
||
int first_p, second_p;
|
||
|
||
/* If both are shorts, return 1; if neither is a short, keep
|
||
checking. */
|
||
first_p = (strstr (first, "short") != NULL);
|
||
second_p = (strstr (second, "short") != NULL);
|
||
if (first_p && second_p)
|
||
return 1;
|
||
if (first_p || second_p)
|
||
return 0;
|
||
|
||
/* Likewise for long. */
|
||
first_p = (strstr (first, "long") != NULL);
|
||
second_p = (strstr (second, "long") != NULL);
|
||
if (first_p && second_p)
|
||
return 1;
|
||
if (first_p || second_p)
|
||
return 0;
|
||
|
||
/* Likewise for char. */
|
||
first_p = (strstr (first, "char") != NULL);
|
||
second_p = (strstr (second, "char") != NULL);
|
||
if (first_p && second_p)
|
||
return 1;
|
||
if (first_p || second_p)
|
||
return 0;
|
||
|
||
/* They must both be ints. */
|
||
return 1;
|
||
}
|
||
|
||
/* Compares type A to type B returns 1 if the represent the same type
|
||
0 otherwise. */
|
||
|
||
int
|
||
types_equal (struct type *a, struct type *b)
|
||
{
|
||
/* Identical type pointers. */
|
||
/* However, this still doesn't catch all cases of same type for b
|
||
and a. The reason is that builtin types are different from
|
||
the same ones constructed from the object. */
|
||
if (a == b)
|
||
return 1;
|
||
|
||
/* Resolve typedefs */
|
||
if (TYPE_CODE (a) == TYPE_CODE_TYPEDEF)
|
||
a = check_typedef (a);
|
||
if (TYPE_CODE (b) == TYPE_CODE_TYPEDEF)
|
||
b = check_typedef (b);
|
||
|
||
/* If after resolving typedefs a and b are not of the same type
|
||
code then they are not equal. */
|
||
if (TYPE_CODE (a) != TYPE_CODE (b))
|
||
return 0;
|
||
|
||
/* If a and b are both pointers types or both reference types then
|
||
they are equal of the same type iff the objects they refer to are
|
||
of the same type. */
|
||
if (TYPE_CODE (a) == TYPE_CODE_PTR
|
||
|| TYPE_CODE (a) == TYPE_CODE_REF)
|
||
return types_equal (TYPE_TARGET_TYPE (a),
|
||
TYPE_TARGET_TYPE (b));
|
||
|
||
/* Well, damnit, if the names are exactly the same, I'll say they
|
||
are exactly the same. This happens when we generate method
|
||
stubs. The types won't point to the same address, but they
|
||
really are the same. */
|
||
|
||
if (TYPE_NAME (a) && TYPE_NAME (b)
|
||
&& strcmp (TYPE_NAME (a), TYPE_NAME (b)) == 0)
|
||
return 1;
|
||
|
||
/* Check if identical after resolving typedefs. */
|
||
if (a == b)
|
||
return 1;
|
||
|
||
/* Two function types are equal if their argument and return types
|
||
are equal. */
|
||
if (TYPE_CODE (a) == TYPE_CODE_FUNC)
|
||
{
|
||
int i;
|
||
|
||
if (TYPE_NFIELDS (a) != TYPE_NFIELDS (b))
|
||
return 0;
|
||
|
||
if (!types_equal (TYPE_TARGET_TYPE (a), TYPE_TARGET_TYPE (b)))
|
||
return 0;
|
||
|
||
for (i = 0; i < TYPE_NFIELDS (a); ++i)
|
||
if (!types_equal (TYPE_FIELD_TYPE (a, i), TYPE_FIELD_TYPE (b, i)))
|
||
return 0;
|
||
|
||
return 1;
|
||
}
|
||
|
||
return 0;
|
||
}
|
||
|
||
/* Deep comparison of types. */
|
||
|
||
/* An entry in the type-equality bcache. */
|
||
|
||
typedef struct type_equality_entry
|
||
{
|
||
struct type *type1, *type2;
|
||
} type_equality_entry_d;
|
||
|
||
DEF_VEC_O (type_equality_entry_d);
|
||
|
||
/* A helper function to compare two strings. Returns 1 if they are
|
||
the same, 0 otherwise. Handles NULLs properly. */
|
||
|
||
static int
|
||
compare_maybe_null_strings (const char *s, const char *t)
|
||
{
|
||
if (s == NULL && t != NULL)
|
||
return 0;
|
||
else if (s != NULL && t == NULL)
|
||
return 0;
|
||
else if (s == NULL && t== NULL)
|
||
return 1;
|
||
return strcmp (s, t) == 0;
|
||
}
|
||
|
||
/* A helper function for check_types_worklist that checks two types for
|
||
"deep" equality. Returns non-zero if the types are considered the
|
||
same, zero otherwise. */
|
||
|
||
static int
|
||
check_types_equal (struct type *type1, struct type *type2,
|
||
VEC (type_equality_entry_d) **worklist)
|
||
{
|
||
type1 = check_typedef (type1);
|
||
type2 = check_typedef (type2);
|
||
|
||
if (type1 == type2)
|
||
return 1;
|
||
|
||
if (TYPE_CODE (type1) != TYPE_CODE (type2)
|
||
|| TYPE_LENGTH (type1) != TYPE_LENGTH (type2)
|
||
|| TYPE_UNSIGNED (type1) != TYPE_UNSIGNED (type2)
|
||
|| TYPE_NOSIGN (type1) != TYPE_NOSIGN (type2)
|
||
|| TYPE_VARARGS (type1) != TYPE_VARARGS (type2)
|
||
|| TYPE_VECTOR (type1) != TYPE_VECTOR (type2)
|
||
|| TYPE_NOTTEXT (type1) != TYPE_NOTTEXT (type2)
|
||
|| TYPE_INSTANCE_FLAGS (type1) != TYPE_INSTANCE_FLAGS (type2)
|
||
|| TYPE_NFIELDS (type1) != TYPE_NFIELDS (type2))
|
||
return 0;
|
||
|
||
if (!compare_maybe_null_strings (TYPE_TAG_NAME (type1),
|
||
TYPE_TAG_NAME (type2)))
|
||
return 0;
|
||
if (!compare_maybe_null_strings (TYPE_NAME (type1), TYPE_NAME (type2)))
|
||
return 0;
|
||
|
||
if (TYPE_CODE (type1) == TYPE_CODE_RANGE)
|
||
{
|
||
if (memcmp (TYPE_RANGE_DATA (type1), TYPE_RANGE_DATA (type2),
|
||
sizeof (*TYPE_RANGE_DATA (type1))) != 0)
|
||
return 0;
|
||
}
|
||
else
|
||
{
|
||
int i;
|
||
|
||
for (i = 0; i < TYPE_NFIELDS (type1); ++i)
|
||
{
|
||
const struct field *field1 = &TYPE_FIELD (type1, i);
|
||
const struct field *field2 = &TYPE_FIELD (type2, i);
|
||
struct type_equality_entry entry;
|
||
|
||
if (FIELD_ARTIFICIAL (*field1) != FIELD_ARTIFICIAL (*field2)
|
||
|| FIELD_BITSIZE (*field1) != FIELD_BITSIZE (*field2)
|
||
|| FIELD_LOC_KIND (*field1) != FIELD_LOC_KIND (*field2))
|
||
return 0;
|
||
if (!compare_maybe_null_strings (FIELD_NAME (*field1),
|
||
FIELD_NAME (*field2)))
|
||
return 0;
|
||
switch (FIELD_LOC_KIND (*field1))
|
||
{
|
||
case FIELD_LOC_KIND_BITPOS:
|
||
if (FIELD_BITPOS (*field1) != FIELD_BITPOS (*field2))
|
||
return 0;
|
||
break;
|
||
case FIELD_LOC_KIND_ENUMVAL:
|
||
if (FIELD_ENUMVAL (*field1) != FIELD_ENUMVAL (*field2))
|
||
return 0;
|
||
break;
|
||
case FIELD_LOC_KIND_PHYSADDR:
|
||
if (FIELD_STATIC_PHYSADDR (*field1)
|
||
!= FIELD_STATIC_PHYSADDR (*field2))
|
||
return 0;
|
||
break;
|
||
case FIELD_LOC_KIND_PHYSNAME:
|
||
if (!compare_maybe_null_strings (FIELD_STATIC_PHYSNAME (*field1),
|
||
FIELD_STATIC_PHYSNAME (*field2)))
|
||
return 0;
|
||
break;
|
||
case FIELD_LOC_KIND_DWARF_BLOCK:
|
||
{
|
||
struct dwarf2_locexpr_baton *block1, *block2;
|
||
|
||
block1 = FIELD_DWARF_BLOCK (*field1);
|
||
block2 = FIELD_DWARF_BLOCK (*field2);
|
||
if (block1->per_cu != block2->per_cu
|
||
|| block1->size != block2->size
|
||
|| memcmp (block1->data, block2->data, block1->size) != 0)
|
||
return 0;
|
||
}
|
||
break;
|
||
default:
|
||
internal_error (__FILE__, __LINE__, _("Unsupported field kind "
|
||
"%d by check_types_equal"),
|
||
FIELD_LOC_KIND (*field1));
|
||
}
|
||
|
||
entry.type1 = FIELD_TYPE (*field1);
|
||
entry.type2 = FIELD_TYPE (*field2);
|
||
VEC_safe_push (type_equality_entry_d, *worklist, &entry);
|
||
}
|
||
}
|
||
|
||
if (TYPE_TARGET_TYPE (type1) != NULL)
|
||
{
|
||
struct type_equality_entry entry;
|
||
|
||
if (TYPE_TARGET_TYPE (type2) == NULL)
|
||
return 0;
|
||
|
||
entry.type1 = TYPE_TARGET_TYPE (type1);
|
||
entry.type2 = TYPE_TARGET_TYPE (type2);
|
||
VEC_safe_push (type_equality_entry_d, *worklist, &entry);
|
||
}
|
||
else if (TYPE_TARGET_TYPE (type2) != NULL)
|
||
return 0;
|
||
|
||
return 1;
|
||
}
|
||
|
||
/* Check types on a worklist for equality. Returns zero if any pair
|
||
is not equal, non-zero if they are all considered equal. */
|
||
|
||
static int
|
||
check_types_worklist (VEC (type_equality_entry_d) **worklist,
|
||
struct bcache *cache)
|
||
{
|
||
while (!VEC_empty (type_equality_entry_d, *worklist))
|
||
{
|
||
struct type_equality_entry entry;
|
||
int added;
|
||
|
||
entry = *VEC_last (type_equality_entry_d, *worklist);
|
||
VEC_pop (type_equality_entry_d, *worklist);
|
||
|
||
/* If the type pair has already been visited, we know it is
|
||
ok. */
|
||
bcache_full (&entry, sizeof (entry), cache, &added);
|
||
if (!added)
|
||
continue;
|
||
|
||
if (check_types_equal (entry.type1, entry.type2, worklist) == 0)
|
||
return 0;
|
||
}
|
||
|
||
return 1;
|
||
}
|
||
|
||
/* Return non-zero if types TYPE1 and TYPE2 are equal, as determined by a
|
||
"deep comparison". Otherwise return zero. */
|
||
|
||
int
|
||
types_deeply_equal (struct type *type1, struct type *type2)
|
||
{
|
||
struct gdb_exception except = exception_none;
|
||
int result = 0;
|
||
struct bcache *cache;
|
||
VEC (type_equality_entry_d) *worklist = NULL;
|
||
struct type_equality_entry entry;
|
||
|
||
gdb_assert (type1 != NULL && type2 != NULL);
|
||
|
||
/* Early exit for the simple case. */
|
||
if (type1 == type2)
|
||
return 1;
|
||
|
||
cache = bcache_xmalloc (NULL, NULL);
|
||
|
||
entry.type1 = type1;
|
||
entry.type2 = type2;
|
||
VEC_safe_push (type_equality_entry_d, worklist, &entry);
|
||
|
||
/* check_types_worklist calls several nested helper functions, some
|
||
of which can raise a GDB exception, so we just check and rethrow
|
||
here. If there is a GDB exception, a comparison is not capable
|
||
(or trusted), so exit. */
|
||
TRY
|
||
{
|
||
result = check_types_worklist (&worklist, cache);
|
||
}
|
||
CATCH (ex, RETURN_MASK_ALL)
|
||
{
|
||
except = ex;
|
||
}
|
||
END_CATCH
|
||
|
||
bcache_xfree (cache);
|
||
VEC_free (type_equality_entry_d, worklist);
|
||
|
||
/* Rethrow if there was a problem. */
|
||
if (except.reason < 0)
|
||
throw_exception (except);
|
||
|
||
return result;
|
||
}
|
||
|
||
/* Compare one type (PARM) for compatibility with another (ARG).
|
||
* PARM is intended to be the parameter type of a function; and
|
||
* ARG is the supplied argument's type. This function tests if
|
||
* the latter can be converted to the former.
|
||
* VALUE is the argument's value or NULL if none (or called recursively)
|
||
*
|
||
* Return 0 if they are identical types;
|
||
* Otherwise, return an integer which corresponds to how compatible
|
||
* PARM is to ARG. The higher the return value, the worse the match.
|
||
* Generally the "bad" conversions are all uniformly assigned a 100. */
|
||
|
||
struct rank
|
||
rank_one_type (struct type *parm, struct type *arg, struct value *value)
|
||
{
|
||
struct rank rank = {0,0};
|
||
|
||
if (types_equal (parm, arg))
|
||
return EXACT_MATCH_BADNESS;
|
||
|
||
/* Resolve typedefs */
|
||
if (TYPE_CODE (parm) == TYPE_CODE_TYPEDEF)
|
||
parm = check_typedef (parm);
|
||
if (TYPE_CODE (arg) == TYPE_CODE_TYPEDEF)
|
||
arg = check_typedef (arg);
|
||
|
||
/* See through references, since we can almost make non-references
|
||
references. */
|
||
if (TYPE_CODE (arg) == TYPE_CODE_REF)
|
||
return (sum_ranks (rank_one_type (parm, TYPE_TARGET_TYPE (arg), NULL),
|
||
REFERENCE_CONVERSION_BADNESS));
|
||
if (TYPE_CODE (parm) == TYPE_CODE_REF)
|
||
return (sum_ranks (rank_one_type (TYPE_TARGET_TYPE (parm), arg, NULL),
|
||
REFERENCE_CONVERSION_BADNESS));
|
||
if (overload_debug)
|
||
/* Debugging only. */
|
||
fprintf_filtered (gdb_stderr,
|
||
"------ Arg is %s [%d], parm is %s [%d]\n",
|
||
TYPE_NAME (arg), TYPE_CODE (arg),
|
||
TYPE_NAME (parm), TYPE_CODE (parm));
|
||
|
||
/* x -> y means arg of type x being supplied for parameter of type y. */
|
||
|
||
switch (TYPE_CODE (parm))
|
||
{
|
||
case TYPE_CODE_PTR:
|
||
switch (TYPE_CODE (arg))
|
||
{
|
||
case TYPE_CODE_PTR:
|
||
|
||
/* Allowed pointer conversions are:
|
||
(a) pointer to void-pointer conversion. */
|
||
if (TYPE_CODE (TYPE_TARGET_TYPE (parm)) == TYPE_CODE_VOID)
|
||
return VOID_PTR_CONVERSION_BADNESS;
|
||
|
||
/* (b) pointer to ancestor-pointer conversion. */
|
||
rank.subrank = distance_to_ancestor (TYPE_TARGET_TYPE (parm),
|
||
TYPE_TARGET_TYPE (arg),
|
||
0);
|
||
if (rank.subrank >= 0)
|
||
return sum_ranks (BASE_PTR_CONVERSION_BADNESS, rank);
|
||
|
||
return INCOMPATIBLE_TYPE_BADNESS;
|
||
case TYPE_CODE_ARRAY:
|
||
if (types_equal (TYPE_TARGET_TYPE (parm),
|
||
TYPE_TARGET_TYPE (arg)))
|
||
return EXACT_MATCH_BADNESS;
|
||
return INCOMPATIBLE_TYPE_BADNESS;
|
||
case TYPE_CODE_FUNC:
|
||
return rank_one_type (TYPE_TARGET_TYPE (parm), arg, NULL);
|
||
case TYPE_CODE_INT:
|
||
if (value != NULL && TYPE_CODE (value_type (value)) == TYPE_CODE_INT)
|
||
{
|
||
if (value_as_long (value) == 0)
|
||
{
|
||
/* Null pointer conversion: allow it to be cast to a pointer.
|
||
[4.10.1 of C++ standard draft n3290] */
|
||
return NULL_POINTER_CONVERSION_BADNESS;
|
||
}
|
||
else
|
||
{
|
||
/* If type checking is disabled, allow the conversion. */
|
||
if (!strict_type_checking)
|
||
return NS_INTEGER_POINTER_CONVERSION_BADNESS;
|
||
}
|
||
}
|
||
/* fall through */
|
||
case TYPE_CODE_ENUM:
|
||
case TYPE_CODE_FLAGS:
|
||
case TYPE_CODE_CHAR:
|
||
case TYPE_CODE_RANGE:
|
||
case TYPE_CODE_BOOL:
|
||
default:
|
||
return INCOMPATIBLE_TYPE_BADNESS;
|
||
}
|
||
case TYPE_CODE_ARRAY:
|
||
switch (TYPE_CODE (arg))
|
||
{
|
||
case TYPE_CODE_PTR:
|
||
case TYPE_CODE_ARRAY:
|
||
return rank_one_type (TYPE_TARGET_TYPE (parm),
|
||
TYPE_TARGET_TYPE (arg), NULL);
|
||
default:
|
||
return INCOMPATIBLE_TYPE_BADNESS;
|
||
}
|
||
case TYPE_CODE_FUNC:
|
||
switch (TYPE_CODE (arg))
|
||
{
|
||
case TYPE_CODE_PTR: /* funcptr -> func */
|
||
return rank_one_type (parm, TYPE_TARGET_TYPE (arg), NULL);
|
||
default:
|
||
return INCOMPATIBLE_TYPE_BADNESS;
|
||
}
|
||
case TYPE_CODE_INT:
|
||
switch (TYPE_CODE (arg))
|
||
{
|
||
case TYPE_CODE_INT:
|
||
if (TYPE_LENGTH (arg) == TYPE_LENGTH (parm))
|
||
{
|
||
/* Deal with signed, unsigned, and plain chars and
|
||
signed and unsigned ints. */
|
||
if (TYPE_NOSIGN (parm))
|
||
{
|
||
/* This case only for character types. */
|
||
if (TYPE_NOSIGN (arg))
|
||
return EXACT_MATCH_BADNESS; /* plain char -> plain char */
|
||
else /* signed/unsigned char -> plain char */
|
||
return INTEGER_CONVERSION_BADNESS;
|
||
}
|
||
else if (TYPE_UNSIGNED (parm))
|
||
{
|
||
if (TYPE_UNSIGNED (arg))
|
||
{
|
||
/* unsigned int -> unsigned int, or
|
||
unsigned long -> unsigned long */
|
||
if (integer_types_same_name_p (TYPE_NAME (parm),
|
||
TYPE_NAME (arg)))
|
||
return EXACT_MATCH_BADNESS;
|
||
else if (integer_types_same_name_p (TYPE_NAME (arg),
|
||
"int")
|
||
&& integer_types_same_name_p (TYPE_NAME (parm),
|
||
"long"))
|
||
/* unsigned int -> unsigned long */
|
||
return INTEGER_PROMOTION_BADNESS;
|
||
else
|
||
/* unsigned long -> unsigned int */
|
||
return INTEGER_CONVERSION_BADNESS;
|
||
}
|
||
else
|
||
{
|
||
if (integer_types_same_name_p (TYPE_NAME (arg),
|
||
"long")
|
||
&& integer_types_same_name_p (TYPE_NAME (parm),
|
||
"int"))
|
||
/* signed long -> unsigned int */
|
||
return INTEGER_CONVERSION_BADNESS;
|
||
else
|
||
/* signed int/long -> unsigned int/long */
|
||
return INTEGER_CONVERSION_BADNESS;
|
||
}
|
||
}
|
||
else if (!TYPE_NOSIGN (arg) && !TYPE_UNSIGNED (arg))
|
||
{
|
||
if (integer_types_same_name_p (TYPE_NAME (parm),
|
||
TYPE_NAME (arg)))
|
||
return EXACT_MATCH_BADNESS;
|
||
else if (integer_types_same_name_p (TYPE_NAME (arg),
|
||
"int")
|
||
&& integer_types_same_name_p (TYPE_NAME (parm),
|
||
"long"))
|
||
return INTEGER_PROMOTION_BADNESS;
|
||
else
|
||
return INTEGER_CONVERSION_BADNESS;
|
||
}
|
||
else
|
||
return INTEGER_CONVERSION_BADNESS;
|
||
}
|
||
else if (TYPE_LENGTH (arg) < TYPE_LENGTH (parm))
|
||
return INTEGER_PROMOTION_BADNESS;
|
||
else
|
||
return INTEGER_CONVERSION_BADNESS;
|
||
case TYPE_CODE_ENUM:
|
||
case TYPE_CODE_FLAGS:
|
||
case TYPE_CODE_CHAR:
|
||
case TYPE_CODE_RANGE:
|
||
case TYPE_CODE_BOOL:
|
||
if (TYPE_DECLARED_CLASS (arg))
|
||
return INCOMPATIBLE_TYPE_BADNESS;
|
||
return INTEGER_PROMOTION_BADNESS;
|
||
case TYPE_CODE_FLT:
|
||
return INT_FLOAT_CONVERSION_BADNESS;
|
||
case TYPE_CODE_PTR:
|
||
return NS_POINTER_CONVERSION_BADNESS;
|
||
default:
|
||
return INCOMPATIBLE_TYPE_BADNESS;
|
||
}
|
||
break;
|
||
case TYPE_CODE_ENUM:
|
||
switch (TYPE_CODE (arg))
|
||
{
|
||
case TYPE_CODE_INT:
|
||
case TYPE_CODE_CHAR:
|
||
case TYPE_CODE_RANGE:
|
||
case TYPE_CODE_BOOL:
|
||
case TYPE_CODE_ENUM:
|
||
if (TYPE_DECLARED_CLASS (parm) || TYPE_DECLARED_CLASS (arg))
|
||
return INCOMPATIBLE_TYPE_BADNESS;
|
||
return INTEGER_CONVERSION_BADNESS;
|
||
case TYPE_CODE_FLT:
|
||
return INT_FLOAT_CONVERSION_BADNESS;
|
||
default:
|
||
return INCOMPATIBLE_TYPE_BADNESS;
|
||
}
|
||
break;
|
||
case TYPE_CODE_CHAR:
|
||
switch (TYPE_CODE (arg))
|
||
{
|
||
case TYPE_CODE_RANGE:
|
||
case TYPE_CODE_BOOL:
|
||
case TYPE_CODE_ENUM:
|
||
if (TYPE_DECLARED_CLASS (arg))
|
||
return INCOMPATIBLE_TYPE_BADNESS;
|
||
return INTEGER_CONVERSION_BADNESS;
|
||
case TYPE_CODE_FLT:
|
||
return INT_FLOAT_CONVERSION_BADNESS;
|
||
case TYPE_CODE_INT:
|
||
if (TYPE_LENGTH (arg) > TYPE_LENGTH (parm))
|
||
return INTEGER_CONVERSION_BADNESS;
|
||
else if (TYPE_LENGTH (arg) < TYPE_LENGTH (parm))
|
||
return INTEGER_PROMOTION_BADNESS;
|
||
/* >>> !! else fall through !! <<< */
|
||
case TYPE_CODE_CHAR:
|
||
/* Deal with signed, unsigned, and plain chars for C++ and
|
||
with int cases falling through from previous case. */
|
||
if (TYPE_NOSIGN (parm))
|
||
{
|
||
if (TYPE_NOSIGN (arg))
|
||
return EXACT_MATCH_BADNESS;
|
||
else
|
||
return INTEGER_CONVERSION_BADNESS;
|
||
}
|
||
else if (TYPE_UNSIGNED (parm))
|
||
{
|
||
if (TYPE_UNSIGNED (arg))
|
||
return EXACT_MATCH_BADNESS;
|
||
else
|
||
return INTEGER_PROMOTION_BADNESS;
|
||
}
|
||
else if (!TYPE_NOSIGN (arg) && !TYPE_UNSIGNED (arg))
|
||
return EXACT_MATCH_BADNESS;
|
||
else
|
||
return INTEGER_CONVERSION_BADNESS;
|
||
default:
|
||
return INCOMPATIBLE_TYPE_BADNESS;
|
||
}
|
||
break;
|
||
case TYPE_CODE_RANGE:
|
||
switch (TYPE_CODE (arg))
|
||
{
|
||
case TYPE_CODE_INT:
|
||
case TYPE_CODE_CHAR:
|
||
case TYPE_CODE_RANGE:
|
||
case TYPE_CODE_BOOL:
|
||
case TYPE_CODE_ENUM:
|
||
return INTEGER_CONVERSION_BADNESS;
|
||
case TYPE_CODE_FLT:
|
||
return INT_FLOAT_CONVERSION_BADNESS;
|
||
default:
|
||
return INCOMPATIBLE_TYPE_BADNESS;
|
||
}
|
||
break;
|
||
case TYPE_CODE_BOOL:
|
||
switch (TYPE_CODE (arg))
|
||
{
|
||
/* n3290 draft, section 4.12.1 (conv.bool):
|
||
|
||
"A prvalue of arithmetic, unscoped enumeration, pointer, or
|
||
pointer to member type can be converted to a prvalue of type
|
||
bool. A zero value, null pointer value, or null member pointer
|
||
value is converted to false; any other value is converted to
|
||
true. A prvalue of type std::nullptr_t can be converted to a
|
||
prvalue of type bool; the resulting value is false." */
|
||
case TYPE_CODE_INT:
|
||
case TYPE_CODE_CHAR:
|
||
case TYPE_CODE_ENUM:
|
||
case TYPE_CODE_FLT:
|
||
case TYPE_CODE_MEMBERPTR:
|
||
case TYPE_CODE_PTR:
|
||
return BOOL_CONVERSION_BADNESS;
|
||
case TYPE_CODE_RANGE:
|
||
return INCOMPATIBLE_TYPE_BADNESS;
|
||
case TYPE_CODE_BOOL:
|
||
return EXACT_MATCH_BADNESS;
|
||
default:
|
||
return INCOMPATIBLE_TYPE_BADNESS;
|
||
}
|
||
break;
|
||
case TYPE_CODE_FLT:
|
||
switch (TYPE_CODE (arg))
|
||
{
|
||
case TYPE_CODE_FLT:
|
||
if (TYPE_LENGTH (arg) < TYPE_LENGTH (parm))
|
||
return FLOAT_PROMOTION_BADNESS;
|
||
else if (TYPE_LENGTH (arg) == TYPE_LENGTH (parm))
|
||
return EXACT_MATCH_BADNESS;
|
||
else
|
||
return FLOAT_CONVERSION_BADNESS;
|
||
case TYPE_CODE_INT:
|
||
case TYPE_CODE_BOOL:
|
||
case TYPE_CODE_ENUM:
|
||
case TYPE_CODE_RANGE:
|
||
case TYPE_CODE_CHAR:
|
||
return INT_FLOAT_CONVERSION_BADNESS;
|
||
default:
|
||
return INCOMPATIBLE_TYPE_BADNESS;
|
||
}
|
||
break;
|
||
case TYPE_CODE_COMPLEX:
|
||
switch (TYPE_CODE (arg))
|
||
{ /* Strictly not needed for C++, but... */
|
||
case TYPE_CODE_FLT:
|
||
return FLOAT_PROMOTION_BADNESS;
|
||
case TYPE_CODE_COMPLEX:
|
||
return EXACT_MATCH_BADNESS;
|
||
default:
|
||
return INCOMPATIBLE_TYPE_BADNESS;
|
||
}
|
||
break;
|
||
case TYPE_CODE_STRUCT:
|
||
switch (TYPE_CODE (arg))
|
||
{
|
||
case TYPE_CODE_STRUCT:
|
||
/* Check for derivation */
|
||
rank.subrank = distance_to_ancestor (parm, arg, 0);
|
||
if (rank.subrank >= 0)
|
||
return sum_ranks (BASE_CONVERSION_BADNESS, rank);
|
||
/* else fall through */
|
||
default:
|
||
return INCOMPATIBLE_TYPE_BADNESS;
|
||
}
|
||
break;
|
||
case TYPE_CODE_UNION:
|
||
switch (TYPE_CODE (arg))
|
||
{
|
||
case TYPE_CODE_UNION:
|
||
default:
|
||
return INCOMPATIBLE_TYPE_BADNESS;
|
||
}
|
||
break;
|
||
case TYPE_CODE_MEMBERPTR:
|
||
switch (TYPE_CODE (arg))
|
||
{
|
||
default:
|
||
return INCOMPATIBLE_TYPE_BADNESS;
|
||
}
|
||
break;
|
||
case TYPE_CODE_METHOD:
|
||
switch (TYPE_CODE (arg))
|
||
{
|
||
|
||
default:
|
||
return INCOMPATIBLE_TYPE_BADNESS;
|
||
}
|
||
break;
|
||
case TYPE_CODE_REF:
|
||
switch (TYPE_CODE (arg))
|
||
{
|
||
|
||
default:
|
||
return INCOMPATIBLE_TYPE_BADNESS;
|
||
}
|
||
|
||
break;
|
||
case TYPE_CODE_SET:
|
||
switch (TYPE_CODE (arg))
|
||
{
|
||
/* Not in C++ */
|
||
case TYPE_CODE_SET:
|
||
return rank_one_type (TYPE_FIELD_TYPE (parm, 0),
|
||
TYPE_FIELD_TYPE (arg, 0), NULL);
|
||
default:
|
||
return INCOMPATIBLE_TYPE_BADNESS;
|
||
}
|
||
break;
|
||
case TYPE_CODE_VOID:
|
||
default:
|
||
return INCOMPATIBLE_TYPE_BADNESS;
|
||
} /* switch (TYPE_CODE (arg)) */
|
||
}
|
||
|
||
/* End of functions for overload resolution. */
|
||
|
||
/* Routines to pretty-print types. */
|
||
|
||
static void
|
||
print_bit_vector (B_TYPE *bits, int nbits)
|
||
{
|
||
int bitno;
|
||
|
||
for (bitno = 0; bitno < nbits; bitno++)
|
||
{
|
||
if ((bitno % 8) == 0)
|
||
{
|
||
puts_filtered (" ");
|
||
}
|
||
if (B_TST (bits, bitno))
|
||
printf_filtered (("1"));
|
||
else
|
||
printf_filtered (("0"));
|
||
}
|
||
}
|
||
|
||
/* Note the first arg should be the "this" pointer, we may not want to
|
||
include it since we may get into a infinitely recursive
|
||
situation. */
|
||
|
||
static void
|
||
print_args (struct field *args, int nargs, int spaces)
|
||
{
|
||
if (args != NULL)
|
||
{
|
||
int i;
|
||
|
||
for (i = 0; i < nargs; i++)
|
||
{
|
||
printfi_filtered (spaces, "[%d] name '%s'\n", i,
|
||
args[i].name != NULL ? args[i].name : "<NULL>");
|
||
recursive_dump_type (args[i].type, spaces + 2);
|
||
}
|
||
}
|
||
}
|
||
|
||
int
|
||
field_is_static (struct field *f)
|
||
{
|
||
/* "static" fields are the fields whose location is not relative
|
||
to the address of the enclosing struct. It would be nice to
|
||
have a dedicated flag that would be set for static fields when
|
||
the type is being created. But in practice, checking the field
|
||
loc_kind should give us an accurate answer. */
|
||
return (FIELD_LOC_KIND (*f) == FIELD_LOC_KIND_PHYSNAME
|
||
|| FIELD_LOC_KIND (*f) == FIELD_LOC_KIND_PHYSADDR);
|
||
}
|
||
|
||
static void
|
||
dump_fn_fieldlists (struct type *type, int spaces)
|
||
{
|
||
int method_idx;
|
||
int overload_idx;
|
||
struct fn_field *f;
|
||
|
||
printfi_filtered (spaces, "fn_fieldlists ");
|
||
gdb_print_host_address (TYPE_FN_FIELDLISTS (type), gdb_stdout);
|
||
printf_filtered ("\n");
|
||
for (method_idx = 0; method_idx < TYPE_NFN_FIELDS (type); method_idx++)
|
||
{
|
||
f = TYPE_FN_FIELDLIST1 (type, method_idx);
|
||
printfi_filtered (spaces + 2, "[%d] name '%s' (",
|
||
method_idx,
|
||
TYPE_FN_FIELDLIST_NAME (type, method_idx));
|
||
gdb_print_host_address (TYPE_FN_FIELDLIST_NAME (type, method_idx),
|
||
gdb_stdout);
|
||
printf_filtered (_(") length %d\n"),
|
||
TYPE_FN_FIELDLIST_LENGTH (type, method_idx));
|
||
for (overload_idx = 0;
|
||
overload_idx < TYPE_FN_FIELDLIST_LENGTH (type, method_idx);
|
||
overload_idx++)
|
||
{
|
||
printfi_filtered (spaces + 4, "[%d] physname '%s' (",
|
||
overload_idx,
|
||
TYPE_FN_FIELD_PHYSNAME (f, overload_idx));
|
||
gdb_print_host_address (TYPE_FN_FIELD_PHYSNAME (f, overload_idx),
|
||
gdb_stdout);
|
||
printf_filtered (")\n");
|
||
printfi_filtered (spaces + 8, "type ");
|
||
gdb_print_host_address (TYPE_FN_FIELD_TYPE (f, overload_idx),
|
||
gdb_stdout);
|
||
printf_filtered ("\n");
|
||
|
||
recursive_dump_type (TYPE_FN_FIELD_TYPE (f, overload_idx),
|
||
spaces + 8 + 2);
|
||
|
||
printfi_filtered (spaces + 8, "args ");
|
||
gdb_print_host_address (TYPE_FN_FIELD_ARGS (f, overload_idx),
|
||
gdb_stdout);
|
||
printf_filtered ("\n");
|
||
print_args (TYPE_FN_FIELD_ARGS (f, overload_idx),
|
||
TYPE_NFIELDS (TYPE_FN_FIELD_TYPE (f, overload_idx)),
|
||
spaces + 8 + 2);
|
||
printfi_filtered (spaces + 8, "fcontext ");
|
||
gdb_print_host_address (TYPE_FN_FIELD_FCONTEXT (f, overload_idx),
|
||
gdb_stdout);
|
||
printf_filtered ("\n");
|
||
|
||
printfi_filtered (spaces + 8, "is_const %d\n",
|
||
TYPE_FN_FIELD_CONST (f, overload_idx));
|
||
printfi_filtered (spaces + 8, "is_volatile %d\n",
|
||
TYPE_FN_FIELD_VOLATILE (f, overload_idx));
|
||
printfi_filtered (spaces + 8, "is_private %d\n",
|
||
TYPE_FN_FIELD_PRIVATE (f, overload_idx));
|
||
printfi_filtered (spaces + 8, "is_protected %d\n",
|
||
TYPE_FN_FIELD_PROTECTED (f, overload_idx));
|
||
printfi_filtered (spaces + 8, "is_stub %d\n",
|
||
TYPE_FN_FIELD_STUB (f, overload_idx));
|
||
printfi_filtered (spaces + 8, "voffset %u\n",
|
||
TYPE_FN_FIELD_VOFFSET (f, overload_idx));
|
||
}
|
||
}
|
||
}
|
||
|
||
static void
|
||
print_cplus_stuff (struct type *type, int spaces)
|
||
{
|
||
printfi_filtered (spaces, "vptr_fieldno %d\n", TYPE_VPTR_FIELDNO (type));
|
||
printfi_filtered (spaces, "vptr_basetype ");
|
||
gdb_print_host_address (TYPE_VPTR_BASETYPE (type), gdb_stdout);
|
||
puts_filtered ("\n");
|
||
if (TYPE_VPTR_BASETYPE (type) != NULL)
|
||
recursive_dump_type (TYPE_VPTR_BASETYPE (type), spaces + 2);
|
||
|
||
printfi_filtered (spaces, "n_baseclasses %d\n",
|
||
TYPE_N_BASECLASSES (type));
|
||
printfi_filtered (spaces, "nfn_fields %d\n",
|
||
TYPE_NFN_FIELDS (type));
|
||
if (TYPE_N_BASECLASSES (type) > 0)
|
||
{
|
||
printfi_filtered (spaces, "virtual_field_bits (%d bits at *",
|
||
TYPE_N_BASECLASSES (type));
|
||
gdb_print_host_address (TYPE_FIELD_VIRTUAL_BITS (type),
|
||
gdb_stdout);
|
||
printf_filtered (")");
|
||
|
||
print_bit_vector (TYPE_FIELD_VIRTUAL_BITS (type),
|
||
TYPE_N_BASECLASSES (type));
|
||
puts_filtered ("\n");
|
||
}
|
||
if (TYPE_NFIELDS (type) > 0)
|
||
{
|
||
if (TYPE_FIELD_PRIVATE_BITS (type) != NULL)
|
||
{
|
||
printfi_filtered (spaces,
|
||
"private_field_bits (%d bits at *",
|
||
TYPE_NFIELDS (type));
|
||
gdb_print_host_address (TYPE_FIELD_PRIVATE_BITS (type),
|
||
gdb_stdout);
|
||
printf_filtered (")");
|
||
print_bit_vector (TYPE_FIELD_PRIVATE_BITS (type),
|
||
TYPE_NFIELDS (type));
|
||
puts_filtered ("\n");
|
||
}
|
||
if (TYPE_FIELD_PROTECTED_BITS (type) != NULL)
|
||
{
|
||
printfi_filtered (spaces,
|
||
"protected_field_bits (%d bits at *",
|
||
TYPE_NFIELDS (type));
|
||
gdb_print_host_address (TYPE_FIELD_PROTECTED_BITS (type),
|
||
gdb_stdout);
|
||
printf_filtered (")");
|
||
print_bit_vector (TYPE_FIELD_PROTECTED_BITS (type),
|
||
TYPE_NFIELDS (type));
|
||
puts_filtered ("\n");
|
||
}
|
||
}
|
||
if (TYPE_NFN_FIELDS (type) > 0)
|
||
{
|
||
dump_fn_fieldlists (type, spaces);
|
||
}
|
||
}
|
||
|
||
/* Print the contents of the TYPE's type_specific union, assuming that
|
||
its type-specific kind is TYPE_SPECIFIC_GNAT_STUFF. */
|
||
|
||
static void
|
||
print_gnat_stuff (struct type *type, int spaces)
|
||
{
|
||
struct type *descriptive_type = TYPE_DESCRIPTIVE_TYPE (type);
|
||
|
||
if (descriptive_type == NULL)
|
||
printfi_filtered (spaces + 2, "no descriptive type\n");
|
||
else
|
||
{
|
||
printfi_filtered (spaces + 2, "descriptive type\n");
|
||
recursive_dump_type (descriptive_type, spaces + 4);
|
||
}
|
||
}
|
||
|
||
static struct obstack dont_print_type_obstack;
|
||
|
||
void
|
||
recursive_dump_type (struct type *type, int spaces)
|
||
{
|
||
int idx;
|
||
|
||
if (spaces == 0)
|
||
obstack_begin (&dont_print_type_obstack, 0);
|
||
|
||
if (TYPE_NFIELDS (type) > 0
|
||
|| (HAVE_CPLUS_STRUCT (type) && TYPE_NFN_FIELDS (type) > 0))
|
||
{
|
||
struct type **first_dont_print
|
||
= (struct type **) obstack_base (&dont_print_type_obstack);
|
||
|
||
int i = (struct type **)
|
||
obstack_next_free (&dont_print_type_obstack) - first_dont_print;
|
||
|
||
while (--i >= 0)
|
||
{
|
||
if (type == first_dont_print[i])
|
||
{
|
||
printfi_filtered (spaces, "type node ");
|
||
gdb_print_host_address (type, gdb_stdout);
|
||
printf_filtered (_(" <same as already seen type>\n"));
|
||
return;
|
||
}
|
||
}
|
||
|
||
obstack_ptr_grow (&dont_print_type_obstack, type);
|
||
}
|
||
|
||
printfi_filtered (spaces, "type node ");
|
||
gdb_print_host_address (type, gdb_stdout);
|
||
printf_filtered ("\n");
|
||
printfi_filtered (spaces, "name '%s' (",
|
||
TYPE_NAME (type) ? TYPE_NAME (type) : "<NULL>");
|
||
gdb_print_host_address (TYPE_NAME (type), gdb_stdout);
|
||
printf_filtered (")\n");
|
||
printfi_filtered (spaces, "tagname '%s' (",
|
||
TYPE_TAG_NAME (type) ? TYPE_TAG_NAME (type) : "<NULL>");
|
||
gdb_print_host_address (TYPE_TAG_NAME (type), gdb_stdout);
|
||
printf_filtered (")\n");
|
||
printfi_filtered (spaces, "code 0x%x ", TYPE_CODE (type));
|
||
switch (TYPE_CODE (type))
|
||
{
|
||
case TYPE_CODE_UNDEF:
|
||
printf_filtered ("(TYPE_CODE_UNDEF)");
|
||
break;
|
||
case TYPE_CODE_PTR:
|
||
printf_filtered ("(TYPE_CODE_PTR)");
|
||
break;
|
||
case TYPE_CODE_ARRAY:
|
||
printf_filtered ("(TYPE_CODE_ARRAY)");
|
||
break;
|
||
case TYPE_CODE_STRUCT:
|
||
printf_filtered ("(TYPE_CODE_STRUCT)");
|
||
break;
|
||
case TYPE_CODE_UNION:
|
||
printf_filtered ("(TYPE_CODE_UNION)");
|
||
break;
|
||
case TYPE_CODE_ENUM:
|
||
printf_filtered ("(TYPE_CODE_ENUM)");
|
||
break;
|
||
case TYPE_CODE_FLAGS:
|
||
printf_filtered ("(TYPE_CODE_FLAGS)");
|
||
break;
|
||
case TYPE_CODE_FUNC:
|
||
printf_filtered ("(TYPE_CODE_FUNC)");
|
||
break;
|
||
case TYPE_CODE_INT:
|
||
printf_filtered ("(TYPE_CODE_INT)");
|
||
break;
|
||
case TYPE_CODE_FLT:
|
||
printf_filtered ("(TYPE_CODE_FLT)");
|
||
break;
|
||
case TYPE_CODE_VOID:
|
||
printf_filtered ("(TYPE_CODE_VOID)");
|
||
break;
|
||
case TYPE_CODE_SET:
|
||
printf_filtered ("(TYPE_CODE_SET)");
|
||
break;
|
||
case TYPE_CODE_RANGE:
|
||
printf_filtered ("(TYPE_CODE_RANGE)");
|
||
break;
|
||
case TYPE_CODE_STRING:
|
||
printf_filtered ("(TYPE_CODE_STRING)");
|
||
break;
|
||
case TYPE_CODE_ERROR:
|
||
printf_filtered ("(TYPE_CODE_ERROR)");
|
||
break;
|
||
case TYPE_CODE_MEMBERPTR:
|
||
printf_filtered ("(TYPE_CODE_MEMBERPTR)");
|
||
break;
|
||
case TYPE_CODE_METHODPTR:
|
||
printf_filtered ("(TYPE_CODE_METHODPTR)");
|
||
break;
|
||
case TYPE_CODE_METHOD:
|
||
printf_filtered ("(TYPE_CODE_METHOD)");
|
||
break;
|
||
case TYPE_CODE_REF:
|
||
printf_filtered ("(TYPE_CODE_REF)");
|
||
break;
|
||
case TYPE_CODE_CHAR:
|
||
printf_filtered ("(TYPE_CODE_CHAR)");
|
||
break;
|
||
case TYPE_CODE_BOOL:
|
||
printf_filtered ("(TYPE_CODE_BOOL)");
|
||
break;
|
||
case TYPE_CODE_COMPLEX:
|
||
printf_filtered ("(TYPE_CODE_COMPLEX)");
|
||
break;
|
||
case TYPE_CODE_TYPEDEF:
|
||
printf_filtered ("(TYPE_CODE_TYPEDEF)");
|
||
break;
|
||
case TYPE_CODE_NAMESPACE:
|
||
printf_filtered ("(TYPE_CODE_NAMESPACE)");
|
||
break;
|
||
default:
|
||
printf_filtered ("(UNKNOWN TYPE CODE)");
|
||
break;
|
||
}
|
||
puts_filtered ("\n");
|
||
printfi_filtered (spaces, "length %d\n", TYPE_LENGTH (type));
|
||
if (TYPE_OBJFILE_OWNED (type))
|
||
{
|
||
printfi_filtered (spaces, "objfile ");
|
||
gdb_print_host_address (TYPE_OWNER (type).objfile, gdb_stdout);
|
||
}
|
||
else
|
||
{
|
||
printfi_filtered (spaces, "gdbarch ");
|
||
gdb_print_host_address (TYPE_OWNER (type).gdbarch, gdb_stdout);
|
||
}
|
||
printf_filtered ("\n");
|
||
printfi_filtered (spaces, "target_type ");
|
||
gdb_print_host_address (TYPE_TARGET_TYPE (type), gdb_stdout);
|
||
printf_filtered ("\n");
|
||
if (TYPE_TARGET_TYPE (type) != NULL)
|
||
{
|
||
recursive_dump_type (TYPE_TARGET_TYPE (type), spaces + 2);
|
||
}
|
||
printfi_filtered (spaces, "pointer_type ");
|
||
gdb_print_host_address (TYPE_POINTER_TYPE (type), gdb_stdout);
|
||
printf_filtered ("\n");
|
||
printfi_filtered (spaces, "reference_type ");
|
||
gdb_print_host_address (TYPE_REFERENCE_TYPE (type), gdb_stdout);
|
||
printf_filtered ("\n");
|
||
printfi_filtered (spaces, "type_chain ");
|
||
gdb_print_host_address (TYPE_CHAIN (type), gdb_stdout);
|
||
printf_filtered ("\n");
|
||
printfi_filtered (spaces, "instance_flags 0x%x",
|
||
TYPE_INSTANCE_FLAGS (type));
|
||
if (TYPE_CONST (type))
|
||
{
|
||
puts_filtered (" TYPE_FLAG_CONST");
|
||
}
|
||
if (TYPE_VOLATILE (type))
|
||
{
|
||
puts_filtered (" TYPE_FLAG_VOLATILE");
|
||
}
|
||
if (TYPE_CODE_SPACE (type))
|
||
{
|
||
puts_filtered (" TYPE_FLAG_CODE_SPACE");
|
||
}
|
||
if (TYPE_DATA_SPACE (type))
|
||
{
|
||
puts_filtered (" TYPE_FLAG_DATA_SPACE");
|
||
}
|
||
if (TYPE_ADDRESS_CLASS_1 (type))
|
||
{
|
||
puts_filtered (" TYPE_FLAG_ADDRESS_CLASS_1");
|
||
}
|
||
if (TYPE_ADDRESS_CLASS_2 (type))
|
||
{
|
||
puts_filtered (" TYPE_FLAG_ADDRESS_CLASS_2");
|
||
}
|
||
if (TYPE_RESTRICT (type))
|
||
{
|
||
puts_filtered (" TYPE_FLAG_RESTRICT");
|
||
}
|
||
if (TYPE_ATOMIC (type))
|
||
{
|
||
puts_filtered (" TYPE_FLAG_ATOMIC");
|
||
}
|
||
puts_filtered ("\n");
|
||
|
||
printfi_filtered (spaces, "flags");
|
||
if (TYPE_UNSIGNED (type))
|
||
{
|
||
puts_filtered (" TYPE_FLAG_UNSIGNED");
|
||
}
|
||
if (TYPE_NOSIGN (type))
|
||
{
|
||
puts_filtered (" TYPE_FLAG_NOSIGN");
|
||
}
|
||
if (TYPE_STUB (type))
|
||
{
|
||
puts_filtered (" TYPE_FLAG_STUB");
|
||
}
|
||
if (TYPE_TARGET_STUB (type))
|
||
{
|
||
puts_filtered (" TYPE_FLAG_TARGET_STUB");
|
||
}
|
||
if (TYPE_STATIC (type))
|
||
{
|
||
puts_filtered (" TYPE_FLAG_STATIC");
|
||
}
|
||
if (TYPE_PROTOTYPED (type))
|
||
{
|
||
puts_filtered (" TYPE_FLAG_PROTOTYPED");
|
||
}
|
||
if (TYPE_INCOMPLETE (type))
|
||
{
|
||
puts_filtered (" TYPE_FLAG_INCOMPLETE");
|
||
}
|
||
if (TYPE_VARARGS (type))
|
||
{
|
||
puts_filtered (" TYPE_FLAG_VARARGS");
|
||
}
|
||
/* This is used for things like AltiVec registers on ppc. Gcc emits
|
||
an attribute for the array type, which tells whether or not we
|
||
have a vector, instead of a regular array. */
|
||
if (TYPE_VECTOR (type))
|
||
{
|
||
puts_filtered (" TYPE_FLAG_VECTOR");
|
||
}
|
||
if (TYPE_FIXED_INSTANCE (type))
|
||
{
|
||
puts_filtered (" TYPE_FIXED_INSTANCE");
|
||
}
|
||
if (TYPE_STUB_SUPPORTED (type))
|
||
{
|
||
puts_filtered (" TYPE_STUB_SUPPORTED");
|
||
}
|
||
if (TYPE_NOTTEXT (type))
|
||
{
|
||
puts_filtered (" TYPE_NOTTEXT");
|
||
}
|
||
puts_filtered ("\n");
|
||
printfi_filtered (spaces, "nfields %d ", TYPE_NFIELDS (type));
|
||
gdb_print_host_address (TYPE_FIELDS (type), gdb_stdout);
|
||
puts_filtered ("\n");
|
||
for (idx = 0; idx < TYPE_NFIELDS (type); idx++)
|
||
{
|
||
if (TYPE_CODE (type) == TYPE_CODE_ENUM)
|
||
printfi_filtered (spaces + 2,
|
||
"[%d] enumval %s type ",
|
||
idx, plongest (TYPE_FIELD_ENUMVAL (type, idx)));
|
||
else
|
||
printfi_filtered (spaces + 2,
|
||
"[%d] bitpos %d bitsize %d type ",
|
||
idx, TYPE_FIELD_BITPOS (type, idx),
|
||
TYPE_FIELD_BITSIZE (type, idx));
|
||
gdb_print_host_address (TYPE_FIELD_TYPE (type, idx), gdb_stdout);
|
||
printf_filtered (" name '%s' (",
|
||
TYPE_FIELD_NAME (type, idx) != NULL
|
||
? TYPE_FIELD_NAME (type, idx)
|
||
: "<NULL>");
|
||
gdb_print_host_address (TYPE_FIELD_NAME (type, idx), gdb_stdout);
|
||
printf_filtered (")\n");
|
||
if (TYPE_FIELD_TYPE (type, idx) != NULL)
|
||
{
|
||
recursive_dump_type (TYPE_FIELD_TYPE (type, idx), spaces + 4);
|
||
}
|
||
}
|
||
if (TYPE_CODE (type) == TYPE_CODE_RANGE)
|
||
{
|
||
printfi_filtered (spaces, "low %s%s high %s%s\n",
|
||
plongest (TYPE_LOW_BOUND (type)),
|
||
TYPE_LOW_BOUND_UNDEFINED (type) ? " (undefined)" : "",
|
||
plongest (TYPE_HIGH_BOUND (type)),
|
||
TYPE_HIGH_BOUND_UNDEFINED (type)
|
||
? " (undefined)" : "");
|
||
}
|
||
|
||
switch (TYPE_SPECIFIC_FIELD (type))
|
||
{
|
||
case TYPE_SPECIFIC_CPLUS_STUFF:
|
||
printfi_filtered (spaces, "cplus_stuff ");
|
||
gdb_print_host_address (TYPE_CPLUS_SPECIFIC (type),
|
||
gdb_stdout);
|
||
puts_filtered ("\n");
|
||
print_cplus_stuff (type, spaces);
|
||
break;
|
||
|
||
case TYPE_SPECIFIC_GNAT_STUFF:
|
||
printfi_filtered (spaces, "gnat_stuff ");
|
||
gdb_print_host_address (TYPE_GNAT_SPECIFIC (type), gdb_stdout);
|
||
puts_filtered ("\n");
|
||
print_gnat_stuff (type, spaces);
|
||
break;
|
||
|
||
case TYPE_SPECIFIC_FLOATFORMAT:
|
||
printfi_filtered (spaces, "floatformat ");
|
||
if (TYPE_FLOATFORMAT (type) == NULL)
|
||
puts_filtered ("(null)");
|
||
else
|
||
{
|
||
puts_filtered ("{ ");
|
||
if (TYPE_FLOATFORMAT (type)[0] == NULL
|
||
|| TYPE_FLOATFORMAT (type)[0]->name == NULL)
|
||
puts_filtered ("(null)");
|
||
else
|
||
puts_filtered (TYPE_FLOATFORMAT (type)[0]->name);
|
||
|
||
puts_filtered (", ");
|
||
if (TYPE_FLOATFORMAT (type)[1] == NULL
|
||
|| TYPE_FLOATFORMAT (type)[1]->name == NULL)
|
||
puts_filtered ("(null)");
|
||
else
|
||
puts_filtered (TYPE_FLOATFORMAT (type)[1]->name);
|
||
|
||
puts_filtered (" }");
|
||
}
|
||
puts_filtered ("\n");
|
||
break;
|
||
|
||
case TYPE_SPECIFIC_FUNC:
|
||
printfi_filtered (spaces, "calling_convention %d\n",
|
||
TYPE_CALLING_CONVENTION (type));
|
||
/* tail_call_list is not printed. */
|
||
break;
|
||
|
||
case TYPE_SPECIFIC_SELF_TYPE:
|
||
printfi_filtered (spaces, "self_type ");
|
||
gdb_print_host_address (TYPE_SELF_TYPE (type), gdb_stdout);
|
||
puts_filtered ("\n");
|
||
break;
|
||
}
|
||
|
||
if (spaces == 0)
|
||
obstack_free (&dont_print_type_obstack, NULL);
|
||
}
|
||
|
||
/* Trivial helpers for the libiberty hash table, for mapping one
|
||
type to another. */
|
||
|
||
struct type_pair
|
||
{
|
||
struct type *old, *newobj;
|
||
};
|
||
|
||
static hashval_t
|
||
type_pair_hash (const void *item)
|
||
{
|
||
const struct type_pair *pair = (const struct type_pair *) item;
|
||
|
||
return htab_hash_pointer (pair->old);
|
||
}
|
||
|
||
static int
|
||
type_pair_eq (const void *item_lhs, const void *item_rhs)
|
||
{
|
||
const struct type_pair *lhs = (const struct type_pair *) item_lhs;
|
||
const struct type_pair *rhs = (const struct type_pair *) item_rhs;
|
||
|
||
return lhs->old == rhs->old;
|
||
}
|
||
|
||
/* Allocate the hash table used by copy_type_recursive to walk
|
||
types without duplicates. We use OBJFILE's obstack, because
|
||
OBJFILE is about to be deleted. */
|
||
|
||
htab_t
|
||
create_copied_types_hash (struct objfile *objfile)
|
||
{
|
||
return htab_create_alloc_ex (1, type_pair_hash, type_pair_eq,
|
||
NULL, &objfile->objfile_obstack,
|
||
hashtab_obstack_allocate,
|
||
dummy_obstack_deallocate);
|
||
}
|
||
|
||
/* Recursively copy (deep copy) a dynamic attribute list of a type. */
|
||
|
||
static struct dynamic_prop_list *
|
||
copy_dynamic_prop_list (struct obstack *objfile_obstack,
|
||
struct dynamic_prop_list *list)
|
||
{
|
||
struct dynamic_prop_list *copy = list;
|
||
struct dynamic_prop_list **node_ptr = ©
|
||
|
||
while (*node_ptr != NULL)
|
||
{
|
||
struct dynamic_prop_list *node_copy;
|
||
|
||
node_copy = ((struct dynamic_prop_list *)
|
||
obstack_copy (objfile_obstack, *node_ptr,
|
||
sizeof (struct dynamic_prop_list)));
|
||
node_copy->prop = (*node_ptr)->prop;
|
||
*node_ptr = node_copy;
|
||
|
||
node_ptr = &node_copy->next;
|
||
}
|
||
|
||
return copy;
|
||
}
|
||
|
||
/* Recursively copy (deep copy) TYPE, if it is associated with
|
||
OBJFILE. Return a new type owned by the gdbarch associated with the type, a
|
||
saved type if we have already visited TYPE (using COPIED_TYPES), or TYPE if
|
||
it is not associated with OBJFILE. */
|
||
|
||
struct type *
|
||
copy_type_recursive (struct objfile *objfile,
|
||
struct type *type,
|
||
htab_t copied_types)
|
||
{
|
||
struct type_pair *stored, pair;
|
||
void **slot;
|
||
struct type *new_type;
|
||
|
||
if (! TYPE_OBJFILE_OWNED (type))
|
||
return type;
|
||
|
||
/* This type shouldn't be pointing to any types in other objfiles;
|
||
if it did, the type might disappear unexpectedly. */
|
||
gdb_assert (TYPE_OBJFILE (type) == objfile);
|
||
|
||
pair.old = type;
|
||
slot = htab_find_slot (copied_types, &pair, INSERT);
|
||
if (*slot != NULL)
|
||
return ((struct type_pair *) *slot)->newobj;
|
||
|
||
new_type = alloc_type_arch (get_type_arch (type));
|
||
|
||
/* We must add the new type to the hash table immediately, in case
|
||
we encounter this type again during a recursive call below. */
|
||
stored = XOBNEW (&objfile->objfile_obstack, struct type_pair);
|
||
stored->old = type;
|
||
stored->newobj = new_type;
|
||
*slot = stored;
|
||
|
||
/* Copy the common fields of types. For the main type, we simply
|
||
copy the entire thing and then update specific fields as needed. */
|
||
*TYPE_MAIN_TYPE (new_type) = *TYPE_MAIN_TYPE (type);
|
||
TYPE_OBJFILE_OWNED (new_type) = 0;
|
||
TYPE_OWNER (new_type).gdbarch = get_type_arch (type);
|
||
|
||
if (TYPE_NAME (type))
|
||
TYPE_NAME (new_type) = xstrdup (TYPE_NAME (type));
|
||
if (TYPE_TAG_NAME (type))
|
||
TYPE_TAG_NAME (new_type) = xstrdup (TYPE_TAG_NAME (type));
|
||
|
||
TYPE_INSTANCE_FLAGS (new_type) = TYPE_INSTANCE_FLAGS (type);
|
||
TYPE_LENGTH (new_type) = TYPE_LENGTH (type);
|
||
|
||
/* Copy the fields. */
|
||
if (TYPE_NFIELDS (type))
|
||
{
|
||
int i, nfields;
|
||
|
||
nfields = TYPE_NFIELDS (type);
|
||
TYPE_FIELDS (new_type) = XCNEWVEC (struct field, nfields);
|
||
for (i = 0; i < nfields; i++)
|
||
{
|
||
TYPE_FIELD_ARTIFICIAL (new_type, i) =
|
||
TYPE_FIELD_ARTIFICIAL (type, i);
|
||
TYPE_FIELD_BITSIZE (new_type, i) = TYPE_FIELD_BITSIZE (type, i);
|
||
if (TYPE_FIELD_TYPE (type, i))
|
||
TYPE_FIELD_TYPE (new_type, i)
|
||
= copy_type_recursive (objfile, TYPE_FIELD_TYPE (type, i),
|
||
copied_types);
|
||
if (TYPE_FIELD_NAME (type, i))
|
||
TYPE_FIELD_NAME (new_type, i) =
|
||
xstrdup (TYPE_FIELD_NAME (type, i));
|
||
switch (TYPE_FIELD_LOC_KIND (type, i))
|
||
{
|
||
case FIELD_LOC_KIND_BITPOS:
|
||
SET_FIELD_BITPOS (TYPE_FIELD (new_type, i),
|
||
TYPE_FIELD_BITPOS (type, i));
|
||
break;
|
||
case FIELD_LOC_KIND_ENUMVAL:
|
||
SET_FIELD_ENUMVAL (TYPE_FIELD (new_type, i),
|
||
TYPE_FIELD_ENUMVAL (type, i));
|
||
break;
|
||
case FIELD_LOC_KIND_PHYSADDR:
|
||
SET_FIELD_PHYSADDR (TYPE_FIELD (new_type, i),
|
||
TYPE_FIELD_STATIC_PHYSADDR (type, i));
|
||
break;
|
||
case FIELD_LOC_KIND_PHYSNAME:
|
||
SET_FIELD_PHYSNAME (TYPE_FIELD (new_type, i),
|
||
xstrdup (TYPE_FIELD_STATIC_PHYSNAME (type,
|
||
i)));
|
||
break;
|
||
default:
|
||
internal_error (__FILE__, __LINE__,
|
||
_("Unexpected type field location kind: %d"),
|
||
TYPE_FIELD_LOC_KIND (type, i));
|
||
}
|
||
}
|
||
}
|
||
|
||
/* For range types, copy the bounds information. */
|
||
if (TYPE_CODE (type) == TYPE_CODE_RANGE)
|
||
{
|
||
TYPE_RANGE_DATA (new_type) = XNEW (struct range_bounds);
|
||
*TYPE_RANGE_DATA (new_type) = *TYPE_RANGE_DATA (type);
|
||
}
|
||
|
||
if (TYPE_DYN_PROP_LIST (type) != NULL)
|
||
TYPE_DYN_PROP_LIST (new_type)
|
||
= copy_dynamic_prop_list (&objfile->objfile_obstack,
|
||
TYPE_DYN_PROP_LIST (type));
|
||
|
||
|
||
/* Copy pointers to other types. */
|
||
if (TYPE_TARGET_TYPE (type))
|
||
TYPE_TARGET_TYPE (new_type) =
|
||
copy_type_recursive (objfile,
|
||
TYPE_TARGET_TYPE (type),
|
||
copied_types);
|
||
|
||
/* Maybe copy the type_specific bits.
|
||
|
||
NOTE drow/2005-12-09: We do not copy the C++-specific bits like
|
||
base classes and methods. There's no fundamental reason why we
|
||
can't, but at the moment it is not needed. */
|
||
|
||
switch (TYPE_SPECIFIC_FIELD (type))
|
||
{
|
||
case TYPE_SPECIFIC_NONE:
|
||
break;
|
||
case TYPE_SPECIFIC_FUNC:
|
||
INIT_FUNC_SPECIFIC (new_type);
|
||
TYPE_CALLING_CONVENTION (new_type) = TYPE_CALLING_CONVENTION (type);
|
||
TYPE_NO_RETURN (new_type) = TYPE_NO_RETURN (type);
|
||
TYPE_TAIL_CALL_LIST (new_type) = NULL;
|
||
break;
|
||
case TYPE_SPECIFIC_FLOATFORMAT:
|
||
TYPE_FLOATFORMAT (new_type) = TYPE_FLOATFORMAT (type);
|
||
break;
|
||
case TYPE_SPECIFIC_CPLUS_STUFF:
|
||
INIT_CPLUS_SPECIFIC (new_type);
|
||
break;
|
||
case TYPE_SPECIFIC_GNAT_STUFF:
|
||
INIT_GNAT_SPECIFIC (new_type);
|
||
break;
|
||
case TYPE_SPECIFIC_SELF_TYPE:
|
||
set_type_self_type (new_type,
|
||
copy_type_recursive (objfile, TYPE_SELF_TYPE (type),
|
||
copied_types));
|
||
break;
|
||
default:
|
||
gdb_assert_not_reached ("bad type_specific_kind");
|
||
}
|
||
|
||
return new_type;
|
||
}
|
||
|
||
/* Make a copy of the given TYPE, except that the pointer & reference
|
||
types are not preserved.
|
||
|
||
This function assumes that the given type has an associated objfile.
|
||
This objfile is used to allocate the new type. */
|
||
|
||
struct type *
|
||
copy_type (const struct type *type)
|
||
{
|
||
struct type *new_type;
|
||
|
||
gdb_assert (TYPE_OBJFILE_OWNED (type));
|
||
|
||
new_type = alloc_type_copy (type);
|
||
TYPE_INSTANCE_FLAGS (new_type) = TYPE_INSTANCE_FLAGS (type);
|
||
TYPE_LENGTH (new_type) = TYPE_LENGTH (type);
|
||
memcpy (TYPE_MAIN_TYPE (new_type), TYPE_MAIN_TYPE (type),
|
||
sizeof (struct main_type));
|
||
if (TYPE_DYN_PROP_LIST (type) != NULL)
|
||
TYPE_DYN_PROP_LIST (new_type)
|
||
= copy_dynamic_prop_list (&TYPE_OBJFILE (type) -> objfile_obstack,
|
||
TYPE_DYN_PROP_LIST (type));
|
||
|
||
return new_type;
|
||
}
|
||
|
||
/* Helper functions to initialize architecture-specific types. */
|
||
|
||
/* Allocate a type structure associated with GDBARCH and set its
|
||
CODE, LENGTH, and NAME fields. */
|
||
|
||
struct type *
|
||
arch_type (struct gdbarch *gdbarch,
|
||
enum type_code code, int length, char *name)
|
||
{
|
||
struct type *type;
|
||
|
||
type = alloc_type_arch (gdbarch);
|
||
TYPE_CODE (type) = code;
|
||
TYPE_LENGTH (type) = length;
|
||
|
||
if (name)
|
||
TYPE_NAME (type) = gdbarch_obstack_strdup (gdbarch, name);
|
||
|
||
return type;
|
||
}
|
||
|
||
/* Allocate a TYPE_CODE_INT type structure associated with GDBARCH.
|
||
BIT is the type size in bits. If UNSIGNED_P is non-zero, set
|
||
the type's TYPE_UNSIGNED flag. NAME is the type name. */
|
||
|
||
struct type *
|
||
arch_integer_type (struct gdbarch *gdbarch,
|
||
int bit, int unsigned_p, char *name)
|
||
{
|
||
struct type *t;
|
||
|
||
t = arch_type (gdbarch, TYPE_CODE_INT, bit / TARGET_CHAR_BIT, name);
|
||
if (unsigned_p)
|
||
TYPE_UNSIGNED (t) = 1;
|
||
if (name && strcmp (name, "char") == 0)
|
||
TYPE_NOSIGN (t) = 1;
|
||
|
||
return t;
|
||
}
|
||
|
||
/* Allocate a TYPE_CODE_CHAR type structure associated with GDBARCH.
|
||
BIT is the type size in bits. If UNSIGNED_P is non-zero, set
|
||
the type's TYPE_UNSIGNED flag. NAME is the type name. */
|
||
|
||
struct type *
|
||
arch_character_type (struct gdbarch *gdbarch,
|
||
int bit, int unsigned_p, char *name)
|
||
{
|
||
struct type *t;
|
||
|
||
t = arch_type (gdbarch, TYPE_CODE_CHAR, bit / TARGET_CHAR_BIT, name);
|
||
if (unsigned_p)
|
||
TYPE_UNSIGNED (t) = 1;
|
||
|
||
return t;
|
||
}
|
||
|
||
/* Allocate a TYPE_CODE_BOOL type structure associated with GDBARCH.
|
||
BIT is the type size in bits. If UNSIGNED_P is non-zero, set
|
||
the type's TYPE_UNSIGNED flag. NAME is the type name. */
|
||
|
||
struct type *
|
||
arch_boolean_type (struct gdbarch *gdbarch,
|
||
int bit, int unsigned_p, char *name)
|
||
{
|
||
struct type *t;
|
||
|
||
t = arch_type (gdbarch, TYPE_CODE_BOOL, bit / TARGET_CHAR_BIT, name);
|
||
if (unsigned_p)
|
||
TYPE_UNSIGNED (t) = 1;
|
||
|
||
return t;
|
||
}
|
||
|
||
/* Allocate a TYPE_CODE_FLT type structure associated with GDBARCH.
|
||
BIT is the type size in bits; if BIT equals -1, the size is
|
||
determined by the floatformat. NAME is the type name. Set the
|
||
TYPE_FLOATFORMAT from FLOATFORMATS. */
|
||
|
||
struct type *
|
||
arch_float_type (struct gdbarch *gdbarch,
|
||
int bit, char *name, const struct floatformat **floatformats)
|
||
{
|
||
struct type *t;
|
||
|
||
if (bit == -1)
|
||
{
|
||
gdb_assert (floatformats != NULL);
|
||
gdb_assert (floatformats[0] != NULL && floatformats[1] != NULL);
|
||
bit = floatformats[0]->totalsize;
|
||
}
|
||
gdb_assert (bit >= 0);
|
||
|
||
t = arch_type (gdbarch, TYPE_CODE_FLT, bit / TARGET_CHAR_BIT, name);
|
||
TYPE_FLOATFORMAT (t) = floatformats;
|
||
return t;
|
||
}
|
||
|
||
/* Allocate a TYPE_CODE_COMPLEX type structure associated with GDBARCH.
|
||
NAME is the type name. TARGET_TYPE is the component float type. */
|
||
|
||
struct type *
|
||
arch_complex_type (struct gdbarch *gdbarch,
|
||
char *name, struct type *target_type)
|
||
{
|
||
struct type *t;
|
||
|
||
t = arch_type (gdbarch, TYPE_CODE_COMPLEX,
|
||
2 * TYPE_LENGTH (target_type), name);
|
||
TYPE_TARGET_TYPE (t) = target_type;
|
||
return t;
|
||
}
|
||
|
||
/* Allocate a TYPE_CODE_FLAGS type structure associated with GDBARCH.
|
||
NAME is the type name. LENGTH is the size of the flag word in bytes. */
|
||
|
||
struct type *
|
||
arch_flags_type (struct gdbarch *gdbarch, char *name, int length)
|
||
{
|
||
int nfields = length * TARGET_CHAR_BIT;
|
||
struct type *type;
|
||
|
||
type = arch_type (gdbarch, TYPE_CODE_FLAGS, length, name);
|
||
TYPE_UNSIGNED (type) = 1;
|
||
TYPE_NFIELDS (type) = nfields;
|
||
TYPE_FIELDS (type)
|
||
= (struct field *) TYPE_ZALLOC (type, nfields * sizeof (struct field));
|
||
|
||
return type;
|
||
}
|
||
|
||
/* Add field to TYPE_CODE_FLAGS type TYPE to indicate the bit at
|
||
position BITPOS is called NAME. */
|
||
|
||
void
|
||
append_flags_type_flag (struct type *type, int bitpos, char *name)
|
||
{
|
||
gdb_assert (TYPE_CODE (type) == TYPE_CODE_FLAGS);
|
||
gdb_assert (bitpos < TYPE_NFIELDS (type));
|
||
gdb_assert (bitpos >= 0);
|
||
|
||
if (name)
|
||
{
|
||
TYPE_FIELD_NAME (type, bitpos) = xstrdup (name);
|
||
SET_FIELD_BITPOS (TYPE_FIELD (type, bitpos), bitpos);
|
||
}
|
||
else
|
||
{
|
||
/* Don't show this field to the user. */
|
||
SET_FIELD_BITPOS (TYPE_FIELD (type, bitpos), -1);
|
||
}
|
||
}
|
||
|
||
/* Allocate a TYPE_CODE_STRUCT or TYPE_CODE_UNION type structure (as
|
||
specified by CODE) associated with GDBARCH. NAME is the type name. */
|
||
|
||
struct type *
|
||
arch_composite_type (struct gdbarch *gdbarch, char *name, enum type_code code)
|
||
{
|
||
struct type *t;
|
||
|
||
gdb_assert (code == TYPE_CODE_STRUCT || code == TYPE_CODE_UNION);
|
||
t = arch_type (gdbarch, code, 0, NULL);
|
||
TYPE_TAG_NAME (t) = name;
|
||
INIT_CPLUS_SPECIFIC (t);
|
||
return t;
|
||
}
|
||
|
||
/* Add new field with name NAME and type FIELD to composite type T.
|
||
Do not set the field's position or adjust the type's length;
|
||
the caller should do so. Return the new field. */
|
||
|
||
struct field *
|
||
append_composite_type_field_raw (struct type *t, char *name,
|
||
struct type *field)
|
||
{
|
||
struct field *f;
|
||
|
||
TYPE_NFIELDS (t) = TYPE_NFIELDS (t) + 1;
|
||
TYPE_FIELDS (t) = XRESIZEVEC (struct field, TYPE_FIELDS (t),
|
||
TYPE_NFIELDS (t));
|
||
f = &(TYPE_FIELDS (t)[TYPE_NFIELDS (t) - 1]);
|
||
memset (f, 0, sizeof f[0]);
|
||
FIELD_TYPE (f[0]) = field;
|
||
FIELD_NAME (f[0]) = name;
|
||
return f;
|
||
}
|
||
|
||
/* Add new field with name NAME and type FIELD to composite type T.
|
||
ALIGNMENT (if non-zero) specifies the minimum field alignment. */
|
||
|
||
void
|
||
append_composite_type_field_aligned (struct type *t, char *name,
|
||
struct type *field, int alignment)
|
||
{
|
||
struct field *f = append_composite_type_field_raw (t, name, field);
|
||
|
||
if (TYPE_CODE (t) == TYPE_CODE_UNION)
|
||
{
|
||
if (TYPE_LENGTH (t) < TYPE_LENGTH (field))
|
||
TYPE_LENGTH (t) = TYPE_LENGTH (field);
|
||
}
|
||
else if (TYPE_CODE (t) == TYPE_CODE_STRUCT)
|
||
{
|
||
TYPE_LENGTH (t) = TYPE_LENGTH (t) + TYPE_LENGTH (field);
|
||
if (TYPE_NFIELDS (t) > 1)
|
||
{
|
||
SET_FIELD_BITPOS (f[0],
|
||
(FIELD_BITPOS (f[-1])
|
||
+ (TYPE_LENGTH (FIELD_TYPE (f[-1]))
|
||
* TARGET_CHAR_BIT)));
|
||
|
||
if (alignment)
|
||
{
|
||
int left;
|
||
|
||
alignment *= TARGET_CHAR_BIT;
|
||
left = FIELD_BITPOS (f[0]) % alignment;
|
||
|
||
if (left)
|
||
{
|
||
SET_FIELD_BITPOS (f[0], FIELD_BITPOS (f[0]) + (alignment - left));
|
||
TYPE_LENGTH (t) += (alignment - left) / TARGET_CHAR_BIT;
|
||
}
|
||
}
|
||
}
|
||
}
|
||
}
|
||
|
||
/* Add new field with name NAME and type FIELD to composite type T. */
|
||
|
||
void
|
||
append_composite_type_field (struct type *t, char *name,
|
||
struct type *field)
|
||
{
|
||
append_composite_type_field_aligned (t, name, field, 0);
|
||
}
|
||
|
||
static struct gdbarch_data *gdbtypes_data;
|
||
|
||
const struct builtin_type *
|
||
builtin_type (struct gdbarch *gdbarch)
|
||
{
|
||
return (const struct builtin_type *) gdbarch_data (gdbarch, gdbtypes_data);
|
||
}
|
||
|
||
static void *
|
||
gdbtypes_post_init (struct gdbarch *gdbarch)
|
||
{
|
||
struct builtin_type *builtin_type
|
||
= GDBARCH_OBSTACK_ZALLOC (gdbarch, struct builtin_type);
|
||
|
||
/* Basic types. */
|
||
builtin_type->builtin_void
|
||
= arch_type (gdbarch, TYPE_CODE_VOID, 1, "void");
|
||
builtin_type->builtin_char
|
||
= arch_integer_type (gdbarch, TARGET_CHAR_BIT,
|
||
!gdbarch_char_signed (gdbarch), "char");
|
||
builtin_type->builtin_signed_char
|
||
= arch_integer_type (gdbarch, TARGET_CHAR_BIT,
|
||
0, "signed char");
|
||
builtin_type->builtin_unsigned_char
|
||
= arch_integer_type (gdbarch, TARGET_CHAR_BIT,
|
||
1, "unsigned char");
|
||
builtin_type->builtin_short
|
||
= arch_integer_type (gdbarch, gdbarch_short_bit (gdbarch),
|
||
0, "short");
|
||
builtin_type->builtin_unsigned_short
|
||
= arch_integer_type (gdbarch, gdbarch_short_bit (gdbarch),
|
||
1, "unsigned short");
|
||
builtin_type->builtin_int
|
||
= arch_integer_type (gdbarch, gdbarch_int_bit (gdbarch),
|
||
0, "int");
|
||
builtin_type->builtin_unsigned_int
|
||
= arch_integer_type (gdbarch, gdbarch_int_bit (gdbarch),
|
||
1, "unsigned int");
|
||
builtin_type->builtin_long
|
||
= arch_integer_type (gdbarch, gdbarch_long_bit (gdbarch),
|
||
0, "long");
|
||
builtin_type->builtin_unsigned_long
|
||
= arch_integer_type (gdbarch, gdbarch_long_bit (gdbarch),
|
||
1, "unsigned long");
|
||
builtin_type->builtin_long_long
|
||
= arch_integer_type (gdbarch, gdbarch_long_long_bit (gdbarch),
|
||
0, "long long");
|
||
builtin_type->builtin_unsigned_long_long
|
||
= arch_integer_type (gdbarch, gdbarch_long_long_bit (gdbarch),
|
||
1, "unsigned long long");
|
||
builtin_type->builtin_float
|
||
= arch_float_type (gdbarch, gdbarch_float_bit (gdbarch),
|
||
"float", gdbarch_float_format (gdbarch));
|
||
builtin_type->builtin_double
|
||
= arch_float_type (gdbarch, gdbarch_double_bit (gdbarch),
|
||
"double", gdbarch_double_format (gdbarch));
|
||
builtin_type->builtin_long_double
|
||
= arch_float_type (gdbarch, gdbarch_long_double_bit (gdbarch),
|
||
"long double", gdbarch_long_double_format (gdbarch));
|
||
builtin_type->builtin_complex
|
||
= arch_complex_type (gdbarch, "complex",
|
||
builtin_type->builtin_float);
|
||
builtin_type->builtin_double_complex
|
||
= arch_complex_type (gdbarch, "double complex",
|
||
builtin_type->builtin_double);
|
||
builtin_type->builtin_string
|
||
= arch_type (gdbarch, TYPE_CODE_STRING, 1, "string");
|
||
builtin_type->builtin_bool
|
||
= arch_type (gdbarch, TYPE_CODE_BOOL, 1, "bool");
|
||
|
||
/* The following three are about decimal floating point types, which
|
||
are 32-bits, 64-bits and 128-bits respectively. */
|
||
builtin_type->builtin_decfloat
|
||
= arch_type (gdbarch, TYPE_CODE_DECFLOAT, 32 / 8, "_Decimal32");
|
||
builtin_type->builtin_decdouble
|
||
= arch_type (gdbarch, TYPE_CODE_DECFLOAT, 64 / 8, "_Decimal64");
|
||
builtin_type->builtin_declong
|
||
= arch_type (gdbarch, TYPE_CODE_DECFLOAT, 128 / 8, "_Decimal128");
|
||
|
||
/* "True" character types. */
|
||
builtin_type->builtin_true_char
|
||
= arch_character_type (gdbarch, TARGET_CHAR_BIT, 0, "true character");
|
||
builtin_type->builtin_true_unsigned_char
|
||
= arch_character_type (gdbarch, TARGET_CHAR_BIT, 1, "true character");
|
||
|
||
/* Fixed-size integer types. */
|
||
builtin_type->builtin_int0
|
||
= arch_integer_type (gdbarch, 0, 0, "int0_t");
|
||
builtin_type->builtin_int8
|
||
= arch_integer_type (gdbarch, 8, 0, "int8_t");
|
||
builtin_type->builtin_uint8
|
||
= arch_integer_type (gdbarch, 8, 1, "uint8_t");
|
||
builtin_type->builtin_int16
|
||
= arch_integer_type (gdbarch, 16, 0, "int16_t");
|
||
builtin_type->builtin_uint16
|
||
= arch_integer_type (gdbarch, 16, 1, "uint16_t");
|
||
builtin_type->builtin_int32
|
||
= arch_integer_type (gdbarch, 32, 0, "int32_t");
|
||
builtin_type->builtin_uint32
|
||
= arch_integer_type (gdbarch, 32, 1, "uint32_t");
|
||
builtin_type->builtin_int64
|
||
= arch_integer_type (gdbarch, 64, 0, "int64_t");
|
||
builtin_type->builtin_uint64
|
||
= arch_integer_type (gdbarch, 64, 1, "uint64_t");
|
||
builtin_type->builtin_int128
|
||
= arch_integer_type (gdbarch, 128, 0, "int128_t");
|
||
builtin_type->builtin_uint128
|
||
= arch_integer_type (gdbarch, 128, 1, "uint128_t");
|
||
TYPE_INSTANCE_FLAGS (builtin_type->builtin_int8) |=
|
||
TYPE_INSTANCE_FLAG_NOTTEXT;
|
||
TYPE_INSTANCE_FLAGS (builtin_type->builtin_uint8) |=
|
||
TYPE_INSTANCE_FLAG_NOTTEXT;
|
||
|
||
/* Wide character types. */
|
||
builtin_type->builtin_char16
|
||
= arch_integer_type (gdbarch, 16, 0, "char16_t");
|
||
builtin_type->builtin_char32
|
||
= arch_integer_type (gdbarch, 32, 0, "char32_t");
|
||
|
||
|
||
/* Default data/code pointer types. */
|
||
builtin_type->builtin_data_ptr
|
||
= lookup_pointer_type (builtin_type->builtin_void);
|
||
builtin_type->builtin_func_ptr
|
||
= lookup_pointer_type (lookup_function_type (builtin_type->builtin_void));
|
||
builtin_type->builtin_func_func
|
||
= lookup_function_type (builtin_type->builtin_func_ptr);
|
||
|
||
/* This type represents a GDB internal function. */
|
||
builtin_type->internal_fn
|
||
= arch_type (gdbarch, TYPE_CODE_INTERNAL_FUNCTION, 0,
|
||
"<internal function>");
|
||
|
||
/* This type represents an xmethod. */
|
||
builtin_type->xmethod
|
||
= arch_type (gdbarch, TYPE_CODE_XMETHOD, 0, "<xmethod>");
|
||
|
||
return builtin_type;
|
||
}
|
||
|
||
/* This set of objfile-based types is intended to be used by symbol
|
||
readers as basic types. */
|
||
|
||
static const struct objfile_data *objfile_type_data;
|
||
|
||
const struct objfile_type *
|
||
objfile_type (struct objfile *objfile)
|
||
{
|
||
struct gdbarch *gdbarch;
|
||
struct objfile_type *objfile_type
|
||
= (struct objfile_type *) objfile_data (objfile, objfile_type_data);
|
||
|
||
if (objfile_type)
|
||
return objfile_type;
|
||
|
||
objfile_type = OBSTACK_CALLOC (&objfile->objfile_obstack,
|
||
1, struct objfile_type);
|
||
|
||
/* Use the objfile architecture to determine basic type properties. */
|
||
gdbarch = get_objfile_arch (objfile);
|
||
|
||
/* Basic types. */
|
||
objfile_type->builtin_void
|
||
= init_type (TYPE_CODE_VOID, 1,
|
||
0,
|
||
"void", objfile);
|
||
|
||
objfile_type->builtin_char
|
||
= init_type (TYPE_CODE_INT, TARGET_CHAR_BIT / TARGET_CHAR_BIT,
|
||
(TYPE_FLAG_NOSIGN
|
||
| (gdbarch_char_signed (gdbarch) ? 0 : TYPE_FLAG_UNSIGNED)),
|
||
"char", objfile);
|
||
objfile_type->builtin_signed_char
|
||
= init_type (TYPE_CODE_INT, TARGET_CHAR_BIT / TARGET_CHAR_BIT,
|
||
0,
|
||
"signed char", objfile);
|
||
objfile_type->builtin_unsigned_char
|
||
= init_type (TYPE_CODE_INT, TARGET_CHAR_BIT / TARGET_CHAR_BIT,
|
||
TYPE_FLAG_UNSIGNED,
|
||
"unsigned char", objfile);
|
||
objfile_type->builtin_short
|
||
= init_type (TYPE_CODE_INT,
|
||
gdbarch_short_bit (gdbarch) / TARGET_CHAR_BIT,
|
||
0, "short", objfile);
|
||
objfile_type->builtin_unsigned_short
|
||
= init_type (TYPE_CODE_INT,
|
||
gdbarch_short_bit (gdbarch) / TARGET_CHAR_BIT,
|
||
TYPE_FLAG_UNSIGNED, "unsigned short", objfile);
|
||
objfile_type->builtin_int
|
||
= init_type (TYPE_CODE_INT,
|
||
gdbarch_int_bit (gdbarch) / TARGET_CHAR_BIT,
|
||
0, "int", objfile);
|
||
objfile_type->builtin_unsigned_int
|
||
= init_type (TYPE_CODE_INT,
|
||
gdbarch_int_bit (gdbarch) / TARGET_CHAR_BIT,
|
||
TYPE_FLAG_UNSIGNED, "unsigned int", objfile);
|
||
objfile_type->builtin_long
|
||
= init_type (TYPE_CODE_INT,
|
||
gdbarch_long_bit (gdbarch) / TARGET_CHAR_BIT,
|
||
0, "long", objfile);
|
||
objfile_type->builtin_unsigned_long
|
||
= init_type (TYPE_CODE_INT,
|
||
gdbarch_long_bit (gdbarch) / TARGET_CHAR_BIT,
|
||
TYPE_FLAG_UNSIGNED, "unsigned long", objfile);
|
||
objfile_type->builtin_long_long
|
||
= init_type (TYPE_CODE_INT,
|
||
gdbarch_long_long_bit (gdbarch) / TARGET_CHAR_BIT,
|
||
0, "long long", objfile);
|
||
objfile_type->builtin_unsigned_long_long
|
||
= init_type (TYPE_CODE_INT,
|
||
gdbarch_long_long_bit (gdbarch) / TARGET_CHAR_BIT,
|
||
TYPE_FLAG_UNSIGNED, "unsigned long long", objfile);
|
||
|
||
objfile_type->builtin_float
|
||
= init_type (TYPE_CODE_FLT,
|
||
gdbarch_float_bit (gdbarch) / TARGET_CHAR_BIT,
|
||
0, "float", objfile);
|
||
TYPE_FLOATFORMAT (objfile_type->builtin_float)
|
||
= gdbarch_float_format (gdbarch);
|
||
objfile_type->builtin_double
|
||
= init_type (TYPE_CODE_FLT,
|
||
gdbarch_double_bit (gdbarch) / TARGET_CHAR_BIT,
|
||
0, "double", objfile);
|
||
TYPE_FLOATFORMAT (objfile_type->builtin_double)
|
||
= gdbarch_double_format (gdbarch);
|
||
objfile_type->builtin_long_double
|
||
= init_type (TYPE_CODE_FLT,
|
||
gdbarch_long_double_bit (gdbarch) / TARGET_CHAR_BIT,
|
||
0, "long double", objfile);
|
||
TYPE_FLOATFORMAT (objfile_type->builtin_long_double)
|
||
= gdbarch_long_double_format (gdbarch);
|
||
|
||
/* This type represents a type that was unrecognized in symbol read-in. */
|
||
objfile_type->builtin_error
|
||
= init_type (TYPE_CODE_ERROR, 0, 0, "<unknown type>", objfile);
|
||
|
||
/* The following set of types is used for symbols with no
|
||
debug information. */
|
||
objfile_type->nodebug_text_symbol
|
||
= init_type (TYPE_CODE_FUNC, 1, 0,
|
||
"<text variable, no debug info>", objfile);
|
||
TYPE_TARGET_TYPE (objfile_type->nodebug_text_symbol)
|
||
= objfile_type->builtin_int;
|
||
objfile_type->nodebug_text_gnu_ifunc_symbol
|
||
= init_type (TYPE_CODE_FUNC, 1, TYPE_FLAG_GNU_IFUNC,
|
||
"<text gnu-indirect-function variable, no debug info>",
|
||
objfile);
|
||
TYPE_TARGET_TYPE (objfile_type->nodebug_text_gnu_ifunc_symbol)
|
||
= objfile_type->nodebug_text_symbol;
|
||
objfile_type->nodebug_got_plt_symbol
|
||
= init_type (TYPE_CODE_PTR, gdbarch_addr_bit (gdbarch) / 8, 0,
|
||
"<text from jump slot in .got.plt, no debug info>",
|
||
objfile);
|
||
TYPE_TARGET_TYPE (objfile_type->nodebug_got_plt_symbol)
|
||
= objfile_type->nodebug_text_symbol;
|
||
objfile_type->nodebug_data_symbol
|
||
= init_type (TYPE_CODE_INT,
|
||
gdbarch_int_bit (gdbarch) / HOST_CHAR_BIT, 0,
|
||
"<data variable, no debug info>", objfile);
|
||
objfile_type->nodebug_unknown_symbol
|
||
= init_type (TYPE_CODE_INT, 1, 0,
|
||
"<variable (not text or data), no debug info>", objfile);
|
||
objfile_type->nodebug_tls_symbol
|
||
= init_type (TYPE_CODE_INT,
|
||
gdbarch_int_bit (gdbarch) / HOST_CHAR_BIT, 0,
|
||
"<thread local variable, no debug info>", objfile);
|
||
|
||
/* NOTE: on some targets, addresses and pointers are not necessarily
|
||
the same.
|
||
|
||
The upshot is:
|
||
- gdb's `struct type' always describes the target's
|
||
representation.
|
||
- gdb's `struct value' objects should always hold values in
|
||
target form.
|
||
- gdb's CORE_ADDR values are addresses in the unified virtual
|
||
address space that the assembler and linker work with. Thus,
|
||
since target_read_memory takes a CORE_ADDR as an argument, it
|
||
can access any memory on the target, even if the processor has
|
||
separate code and data address spaces.
|
||
|
||
In this context, objfile_type->builtin_core_addr is a bit odd:
|
||
it's a target type for a value the target will never see. It's
|
||
only used to hold the values of (typeless) linker symbols, which
|
||
are indeed in the unified virtual address space. */
|
||
|
||
objfile_type->builtin_core_addr
|
||
= init_type (TYPE_CODE_INT,
|
||
gdbarch_addr_bit (gdbarch) / 8,
|
||
TYPE_FLAG_UNSIGNED, "__CORE_ADDR", objfile);
|
||
|
||
set_objfile_data (objfile, objfile_type_data, objfile_type);
|
||
return objfile_type;
|
||
}
|
||
|
||
extern initialize_file_ftype _initialize_gdbtypes;
|
||
|
||
void
|
||
_initialize_gdbtypes (void)
|
||
{
|
||
gdbtypes_data = gdbarch_data_register_post_init (gdbtypes_post_init);
|
||
objfile_type_data = register_objfile_data ();
|
||
|
||
add_setshow_zuinteger_cmd ("overload", no_class, &overload_debug,
|
||
_("Set debugging of C++ overloading."),
|
||
_("Show debugging of C++ overloading."),
|
||
_("When enabled, ranking of the "
|
||
"functions is displayed."),
|
||
NULL,
|
||
show_overload_debug,
|
||
&setdebuglist, &showdebuglist);
|
||
|
||
/* Add user knob for controlling resolution of opaque types. */
|
||
add_setshow_boolean_cmd ("opaque-type-resolution", class_support,
|
||
&opaque_type_resolution,
|
||
_("Set resolution of opaque struct/class/union"
|
||
" types (if set before loading symbols)."),
|
||
_("Show resolution of opaque struct/class/union"
|
||
" types (if set before loading symbols)."),
|
||
NULL, NULL,
|
||
show_opaque_type_resolution,
|
||
&setlist, &showlist);
|
||
|
||
/* Add an option to permit non-strict type checking. */
|
||
add_setshow_boolean_cmd ("type", class_support,
|
||
&strict_type_checking,
|
||
_("Set strict type checking."),
|
||
_("Show strict type checking."),
|
||
NULL, NULL,
|
||
show_strict_type_checking,
|
||
&setchecklist, &showchecklist);
|
||
}
|