1cdfc98a99
On the testcase in the PR with -fno-tree-sink -O3 -fPIC -fomit-frame-pointer -fno-strict-aliasing -mstackrealign we have prologue: 0000000000000000 <_func_with_dwarf_issue_>: 0: 4c 8d 54 24 08 lea 0x8(%rsp),%r10 5: 48 83 e4 f0 and $0xfffffffffffffff0,%rsp 9: 41 ff 72 f8 pushq -0x8(%r10) d: 55 push %rbp e: 48 89 e5 mov %rsp,%rbp 11: 41 57 push %r15 13: 41 56 push %r14 15: 41 55 push %r13 17: 41 54 push %r12 19: 41 52 push %r10 1b: 53 push %rbx 1c: 48 83 ec 20 sub $0x20,%rsp and emit 00000000 0000000000000014 00000000 CIE Version: 1 Augmentation: "zR" Code alignment factor: 1 Data alignment factor: -8 Return address column: 16 Augmentation data: 1b DW_CFA_def_cfa: r7 (rsp) ofs 8 DW_CFA_offset: r16 (rip) at cfa-8 DW_CFA_nop DW_CFA_nop 00000018 0000000000000044 0000001c FDE cie=00000000 pc=0000000000000000..00000000000001d5 DW_CFA_advance_loc: 5 to 0000000000000005 DW_CFA_def_cfa: r10 (r10) ofs 0 DW_CFA_advance_loc: 9 to 000000000000000e DW_CFA_expression: r6 (rbp) (DW_OP_breg6 (rbp): 0) DW_CFA_advance_loc: 13 to 000000000000001b DW_CFA_def_cfa_expression (DW_OP_breg6 (rbp): -40; DW_OP_deref) DW_CFA_expression: r15 (r15) (DW_OP_breg6 (rbp): -8) DW_CFA_expression: r14 (r14) (DW_OP_breg6 (rbp): -16) DW_CFA_expression: r13 (r13) (DW_OP_breg6 (rbp): -24) DW_CFA_expression: r12 (r12) (DW_OP_breg6 (rbp): -32) ... unwind info for that. The problem is when async signal (or stepping through in the debugger) stops after the pushq %rbp instruction and before movq %rsp, %rbp, the unwind info says that caller's %rbp is saved there at *%rbp, but that is not true, caller's %rbp is either still available in the %rbp register, or in *%rsp, only after executing the next instruction - movq %rsp, %rbp - the location for %rbp is correct. So, either we'd need to temporarily say: DW_CFA_advance_loc: 9 to 000000000000000e DW_CFA_expression: r6 (rbp) (DW_OP_breg7 (rsp): 0) DW_CFA_advance_loc: 3 to 0000000000000011 DW_CFA_expression: r6 (rbp) (DW_OP_breg6 (rbp): 0) DW_CFA_advance_loc: 10 to 000000000000001b or to me it seems more compact to just say: DW_CFA_advance_loc: 12 to 0000000000000011 DW_CFA_expression: r6 (rbp) (DW_OP_breg6 (rbp): 0) DW_CFA_advance_loc: 10 to 000000000000001b I've tried instead to deal with it through REG_FRAME_RELATED_EXPR from the backend, but that failed miserably as explained in the PR, dwarf2cfi.c has some rules (Rule 16 to Rule 19) that are specific to the dynamic stack realignment using drap register that only the i386 backend does right now, and by using REG_FRAME_RELATED_EXPR or REG_CFA* notes we can't emulate those rules. The following patch instead does the deferring of the hard frame pointer save rule in dwarf2cfi.c Rule 18 handling and emits it on the (set hfp sp) assignment that must appear shortly after it and adds assertion that it is the case. The difference before/after the patch on the assembly is: --- pr99334.s~ 2021-03-26 15:42:40.881749380 +0100 +++ pr99334.s 2021-03-26 17:38:05.729161910 +0100 @@ -11,8 +11,8 @@ _func_with_dwarf_issue_: andq $-16, %rsp pushq -8(%r10) pushq %rbp - .cfi_escape 0x10,0x6,0x2,0x76,0 movq %rsp, %rbp + .cfi_escape 0x10,0x6,0x2,0x76,0 pushq %r15 pushq %r14 pushq %r13 i.e. does just what we IMHO need, after pushq %rbp %rbp still contains parent's frame value and so the save rule doesn't need to be overridden there, ditto at the start of the next insn before the side-effect took effect, and we override it only after it when %rbp already has the right value. If some other target adds dynamic stack realignment in the future and the offset 0 case wouldn't be true there, the code can be adjusted so that it works on all the drap architectures, I'm pretty sure the code would need other adjustments too. For the rule 18 and for the (set hfp sp) after it we already have asserts for the drap cases that check whether the code looks the way i?86/x86_64 emit it currently. 2021-03-26 Jakub Jelinek <jakub@redhat.com> PR debug/99334 * dwarf2out.h (struct dw_fde_node): Add rule18 member. * dwarf2cfi.c (dwarf2out_frame_debug_expr): When handling (set hfp sp) assignment with drap_reg active, queue reg save for hfp with offset 0 and flush queued reg saves. When handling a push with rule18, defer queueing reg save for hfp and just assert the offset is 0. (scan_trace): Assert that fde->rule18 is false.
389 lines
13 KiB
C
389 lines
13 KiB
C
/* dwarf2out.h - Various declarations for functions found in dwarf2out.c
|
|
Copyright (C) 1998-2021 Free Software Foundation, Inc.
|
|
|
|
This file is part of GCC.
|
|
|
|
GCC 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, or (at your option) any later
|
|
version.
|
|
|
|
GCC 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 GCC; see the file COPYING3. If not see
|
|
<http://www.gnu.org/licenses/>. */
|
|
|
|
#ifndef GCC_DWARF2OUT_H
|
|
#define GCC_DWARF2OUT_H 1
|
|
|
|
#include "dwarf2.h" /* ??? Remove this once only used by dwarf2foo.c. */
|
|
|
|
typedef struct die_struct *dw_die_ref;
|
|
typedef const struct die_struct *const_dw_die_ref;
|
|
|
|
typedef struct dw_val_node *dw_val_ref;
|
|
typedef struct dw_cfi_node *dw_cfi_ref;
|
|
typedef struct dw_loc_descr_node *dw_loc_descr_ref;
|
|
typedef struct dw_loc_list_struct *dw_loc_list_ref;
|
|
typedef struct dw_discr_list_node *dw_discr_list_ref;
|
|
typedef wide_int *wide_int_ptr;
|
|
|
|
|
|
/* Call frames are described using a sequence of Call Frame
|
|
Information instructions. The register number, offset
|
|
and address fields are provided as possible operands;
|
|
their use is selected by the opcode field. */
|
|
|
|
enum dw_cfi_oprnd_type {
|
|
dw_cfi_oprnd_unused,
|
|
dw_cfi_oprnd_reg_num,
|
|
dw_cfi_oprnd_offset,
|
|
dw_cfi_oprnd_addr,
|
|
dw_cfi_oprnd_loc,
|
|
dw_cfi_oprnd_cfa_loc
|
|
};
|
|
|
|
typedef union GTY(()) {
|
|
unsigned int GTY ((tag ("dw_cfi_oprnd_reg_num"))) dw_cfi_reg_num;
|
|
HOST_WIDE_INT GTY ((tag ("dw_cfi_oprnd_offset"))) dw_cfi_offset;
|
|
const char * GTY ((tag ("dw_cfi_oprnd_addr"))) dw_cfi_addr;
|
|
struct dw_loc_descr_node * GTY ((tag ("dw_cfi_oprnd_loc"))) dw_cfi_loc;
|
|
struct dw_cfa_location * GTY ((tag ("dw_cfi_oprnd_cfa_loc")))
|
|
dw_cfi_cfa_loc;
|
|
} dw_cfi_oprnd;
|
|
|
|
struct GTY(()) dw_cfi_node {
|
|
enum dwarf_call_frame_info dw_cfi_opc;
|
|
dw_cfi_oprnd GTY ((desc ("dw_cfi_oprnd1_desc (%1.dw_cfi_opc)")))
|
|
dw_cfi_oprnd1;
|
|
dw_cfi_oprnd GTY ((desc ("dw_cfi_oprnd2_desc (%1.dw_cfi_opc)")))
|
|
dw_cfi_oprnd2;
|
|
};
|
|
|
|
|
|
typedef vec<dw_cfi_ref, va_gc> *cfi_vec;
|
|
|
|
typedef struct dw_fde_node *dw_fde_ref;
|
|
|
|
/* All call frame descriptions (FDE's) in the GCC generated DWARF
|
|
refer to a single Common Information Entry (CIE), defined at
|
|
the beginning of the .debug_frame section. This use of a single
|
|
CIE obviates the need to keep track of multiple CIE's
|
|
in the DWARF generation routines below. */
|
|
|
|
struct GTY(()) dw_fde_node {
|
|
tree decl;
|
|
const char *dw_fde_begin;
|
|
const char *dw_fde_current_label;
|
|
const char *dw_fde_end;
|
|
const char *dw_fde_vms_end_prologue;
|
|
const char *dw_fde_vms_begin_epilogue;
|
|
const char *dw_fde_second_begin;
|
|
const char *dw_fde_second_end;
|
|
cfi_vec dw_fde_cfi;
|
|
int dw_fde_switch_cfi_index; /* Last CFI before switching sections. */
|
|
HOST_WIDE_INT stack_realignment;
|
|
|
|
unsigned funcdef_number;
|
|
unsigned fde_index;
|
|
|
|
/* Dynamic realign argument pointer register. */
|
|
unsigned int drap_reg;
|
|
/* Virtual dynamic realign argument pointer register. */
|
|
unsigned int vdrap_reg;
|
|
/* These 3 flags are copied from rtl_data in function.h. */
|
|
unsigned all_throwers_are_sibcalls : 1;
|
|
unsigned uses_eh_lsda : 1;
|
|
unsigned nothrow : 1;
|
|
/* Whether we did stack realign in this call frame. */
|
|
unsigned stack_realign : 1;
|
|
/* Whether dynamic realign argument pointer register has been saved. */
|
|
unsigned drap_reg_saved: 1;
|
|
/* True iff dw_fde_begin label is in text_section or cold_text_section. */
|
|
unsigned in_std_section : 1;
|
|
/* True iff dw_fde_second_begin label is in text_section or
|
|
cold_text_section. */
|
|
unsigned second_in_std_section : 1;
|
|
/* True if Rule 18 described in dwarf2cfi.c is in action, i.e. for dynamic
|
|
stack realignment in between pushing of hard frame pointer to stack
|
|
and setting hard frame pointer to stack pointer. The register save for
|
|
hard frame pointer register should be emitted only on the latter
|
|
instruction. */
|
|
unsigned rule18 : 1;
|
|
};
|
|
|
|
|
|
/* This is how we define the location of the CFA. We use to handle it
|
|
as REG + OFFSET all the time, but now it can be more complex.
|
|
It can now be either REG + CFA_OFFSET or *(REG + BASE_OFFSET) + CFA_OFFSET.
|
|
Instead of passing around REG and OFFSET, we pass a copy
|
|
of this structure. */
|
|
struct GTY(()) dw_cfa_location {
|
|
poly_int64_pod offset;
|
|
poly_int64_pod base_offset;
|
|
/* REG is in DWARF_FRAME_REGNUM space, *not* normal REGNO space. */
|
|
unsigned int reg;
|
|
BOOL_BITFIELD indirect : 1; /* 1 if CFA is accessed via a dereference. */
|
|
BOOL_BITFIELD in_use : 1; /* 1 if a saved cfa is stored here. */
|
|
};
|
|
|
|
|
|
/* Each DIE may have a series of attribute/value pairs. Values
|
|
can take on several forms. The forms that are used in this
|
|
implementation are listed below. */
|
|
|
|
enum dw_val_class
|
|
{
|
|
dw_val_class_none,
|
|
dw_val_class_addr,
|
|
dw_val_class_offset,
|
|
dw_val_class_loc,
|
|
dw_val_class_loc_list,
|
|
dw_val_class_range_list,
|
|
dw_val_class_const,
|
|
dw_val_class_unsigned_const,
|
|
dw_val_class_const_double,
|
|
dw_val_class_wide_int,
|
|
dw_val_class_vec,
|
|
dw_val_class_flag,
|
|
dw_val_class_die_ref,
|
|
dw_val_class_fde_ref,
|
|
dw_val_class_lbl_id,
|
|
dw_val_class_lineptr,
|
|
dw_val_class_str,
|
|
dw_val_class_macptr,
|
|
dw_val_class_loclistsptr,
|
|
dw_val_class_file,
|
|
dw_val_class_data8,
|
|
dw_val_class_decl_ref,
|
|
dw_val_class_vms_delta,
|
|
dw_val_class_high_pc,
|
|
dw_val_class_discr_value,
|
|
dw_val_class_discr_list,
|
|
dw_val_class_const_implicit,
|
|
dw_val_class_unsigned_const_implicit,
|
|
dw_val_class_file_implicit,
|
|
dw_val_class_view_list,
|
|
dw_val_class_symview
|
|
};
|
|
|
|
/* Describe a floating point constant value, or a vector constant value. */
|
|
|
|
struct GTY(()) dw_vec_const {
|
|
void * GTY((atomic)) array;
|
|
unsigned length;
|
|
unsigned elt_size;
|
|
};
|
|
|
|
/* Describe a single value that a discriminant can match.
|
|
|
|
Discriminants (in the "record variant part" meaning) are scalars.
|
|
dw_discr_list_ref and dw_discr_value are a mean to describe a set of
|
|
discriminant values that are matched by a particular variant.
|
|
|
|
Discriminants can be signed or unsigned scalars, and can be discriminants
|
|
values. Both have to be consistent, though. */
|
|
|
|
struct GTY(()) dw_discr_value {
|
|
int pos; /* Whether the discriminant value is positive (unsigned). */
|
|
union
|
|
{
|
|
HOST_WIDE_INT GTY ((tag ("0"))) sval;
|
|
unsigned HOST_WIDE_INT GTY ((tag ("1"))) uval;
|
|
}
|
|
GTY ((desc ("%1.pos"))) v;
|
|
};
|
|
|
|
struct addr_table_entry;
|
|
|
|
/* The dw_val_node describes an attribute's value, as it is
|
|
represented internally. */
|
|
|
|
struct GTY(()) dw_val_node {
|
|
enum dw_val_class val_class;
|
|
struct addr_table_entry * GTY(()) val_entry;
|
|
union dw_val_struct_union
|
|
{
|
|
rtx GTY ((tag ("dw_val_class_addr"))) val_addr;
|
|
unsigned HOST_WIDE_INT GTY ((tag ("dw_val_class_offset"))) val_offset;
|
|
dw_loc_list_ref GTY ((tag ("dw_val_class_loc_list"))) val_loc_list;
|
|
dw_die_ref GTY ((tag ("dw_val_class_view_list"))) val_view_list;
|
|
dw_loc_descr_ref GTY ((tag ("dw_val_class_loc"))) val_loc;
|
|
HOST_WIDE_INT GTY ((default)) val_int;
|
|
unsigned HOST_WIDE_INT
|
|
GTY ((tag ("dw_val_class_unsigned_const"))) val_unsigned;
|
|
double_int GTY ((tag ("dw_val_class_const_double"))) val_double;
|
|
wide_int_ptr GTY ((tag ("dw_val_class_wide_int"))) val_wide;
|
|
dw_vec_const GTY ((tag ("dw_val_class_vec"))) val_vec;
|
|
struct dw_val_die_union
|
|
{
|
|
dw_die_ref die;
|
|
int external;
|
|
} GTY ((tag ("dw_val_class_die_ref"))) val_die_ref;
|
|
unsigned GTY ((tag ("dw_val_class_fde_ref"))) val_fde_index;
|
|
struct indirect_string_node * GTY ((tag ("dw_val_class_str"))) val_str;
|
|
char * GTY ((tag ("dw_val_class_lbl_id"))) val_lbl_id;
|
|
unsigned char GTY ((tag ("dw_val_class_flag"))) val_flag;
|
|
struct dwarf_file_data * GTY ((tag ("dw_val_class_file"))) val_file;
|
|
struct dwarf_file_data *
|
|
GTY ((tag ("dw_val_class_file_implicit"))) val_file_implicit;
|
|
unsigned char GTY ((tag ("dw_val_class_data8"))) val_data8[8];
|
|
tree GTY ((tag ("dw_val_class_decl_ref"))) val_decl_ref;
|
|
struct dw_val_vms_delta_union
|
|
{
|
|
char * lbl1;
|
|
char * lbl2;
|
|
} GTY ((tag ("dw_val_class_vms_delta"))) val_vms_delta;
|
|
dw_discr_value GTY ((tag ("dw_val_class_discr_value"))) val_discr_value;
|
|
dw_discr_list_ref GTY ((tag ("dw_val_class_discr_list"))) val_discr_list;
|
|
char * GTY ((tag ("dw_val_class_symview"))) val_symbolic_view;
|
|
}
|
|
GTY ((desc ("%1.val_class"))) v;
|
|
};
|
|
|
|
/* Locations in memory are described using a sequence of stack machine
|
|
operations. */
|
|
|
|
struct GTY((chain_next ("%h.dw_loc_next"))) dw_loc_descr_node {
|
|
dw_loc_descr_ref dw_loc_next;
|
|
ENUM_BITFIELD (dwarf_location_atom) dw_loc_opc : 8;
|
|
/* Used to distinguish DW_OP_addr with a direct symbol relocation
|
|
from DW_OP_addr with a dtp-relative symbol relocation. */
|
|
unsigned int dtprel : 1;
|
|
/* For DW_OP_pick, DW_OP_dup and DW_OP_over operations: true iff.
|
|
it targets a DWARF prodecure argument. In this case, it needs to be
|
|
relocated according to the current frame offset. */
|
|
unsigned int frame_offset_rel : 1;
|
|
int dw_loc_addr;
|
|
dw_val_node dw_loc_oprnd1;
|
|
dw_val_node dw_loc_oprnd2;
|
|
};
|
|
|
|
/* A variant (inside a record variant part) is selected when the corresponding
|
|
discriminant matches its set of values (see the comment for dw_discr_value).
|
|
The following datastructure holds such matching information. */
|
|
|
|
struct GTY(()) dw_discr_list_node {
|
|
dw_discr_list_ref dw_discr_next;
|
|
|
|
dw_discr_value dw_discr_lower_bound;
|
|
dw_discr_value dw_discr_upper_bound;
|
|
/* This node represents only the value in dw_discr_lower_bound when it's
|
|
zero. It represents the range between the two fields (bounds included)
|
|
otherwise. */
|
|
int dw_discr_range;
|
|
};
|
|
|
|
/* Interface from dwarf2out.c to dwarf2cfi.c. */
|
|
extern struct dw_loc_descr_node *build_cfa_loc
|
|
(dw_cfa_location *, poly_int64);
|
|
extern struct dw_loc_descr_node *build_cfa_aligned_loc
|
|
(dw_cfa_location *, poly_int64, HOST_WIDE_INT);
|
|
extern struct dw_loc_descr_node *mem_loc_descriptor
|
|
(rtx, machine_mode mode, machine_mode mem_mode,
|
|
enum var_init_status);
|
|
extern bool loc_descr_equal_p (dw_loc_descr_ref, dw_loc_descr_ref);
|
|
extern dw_fde_ref dwarf2out_alloc_current_fde (void);
|
|
|
|
extern unsigned long size_of_locs (dw_loc_descr_ref);
|
|
extern void output_loc_sequence (dw_loc_descr_ref, int);
|
|
extern void output_loc_sequence_raw (dw_loc_descr_ref);
|
|
|
|
/* Interface from dwarf2cfi.c to dwarf2out.c. */
|
|
extern void lookup_cfa_1 (dw_cfi_ref cfi, dw_cfa_location *loc,
|
|
dw_cfa_location *remember);
|
|
extern bool cfa_equal_p (const dw_cfa_location *, const dw_cfa_location *);
|
|
|
|
extern void output_cfi (dw_cfi_ref, dw_fde_ref, int);
|
|
|
|
extern GTY(()) cfi_vec cie_cfi_vec;
|
|
|
|
/* Interface from dwarf2*.c to the rest of the compiler. */
|
|
extern enum dw_cfi_oprnd_type dw_cfi_oprnd1_desc
|
|
(enum dwarf_call_frame_info cfi);
|
|
extern enum dw_cfi_oprnd_type dw_cfi_oprnd2_desc
|
|
(enum dwarf_call_frame_info cfi);
|
|
|
|
extern void output_cfi_directive (FILE *f, struct dw_cfi_node *cfi);
|
|
|
|
extern void dwarf2out_emit_cfi (dw_cfi_ref cfi);
|
|
|
|
extern void debug_dwarf (void);
|
|
struct die_struct;
|
|
extern void debug_dwarf_die (struct die_struct *);
|
|
extern void debug_dwarf_loc_descr (dw_loc_descr_ref);
|
|
extern void debug (die_struct &ref);
|
|
extern void debug (die_struct *ptr);
|
|
extern void dwarf2out_set_demangle_name_func (const char *(*) (const char *));
|
|
#ifdef VMS_DEBUGGING_INFO
|
|
extern void dwarf2out_vms_debug_main_pointer (void);
|
|
#endif
|
|
|
|
enum array_descr_ordering
|
|
{
|
|
array_descr_ordering_default,
|
|
array_descr_ordering_row_major,
|
|
array_descr_ordering_column_major
|
|
};
|
|
|
|
#define DWARF2OUT_ARRAY_DESCR_INFO_MAX_DIMEN 16
|
|
|
|
struct array_descr_info
|
|
{
|
|
int ndimensions;
|
|
enum array_descr_ordering ordering;
|
|
tree element_type;
|
|
tree base_decl;
|
|
tree data_location;
|
|
tree allocated;
|
|
tree associated;
|
|
tree stride;
|
|
tree rank;
|
|
bool stride_in_bits;
|
|
struct array_descr_dimen
|
|
{
|
|
/* GCC uses sizetype for array indices, so lower_bound and upper_bound
|
|
will likely be "sizetype" values. However, bounds may have another
|
|
type in the original source code. */
|
|
tree bounds_type;
|
|
tree lower_bound;
|
|
tree upper_bound;
|
|
|
|
/* Only Fortran uses more than one dimension for array types. For other
|
|
languages, the stride can be rather specified for the whole array. */
|
|
tree stride;
|
|
} dimen[DWARF2OUT_ARRAY_DESCR_INFO_MAX_DIMEN];
|
|
};
|
|
|
|
enum fixed_point_scale_factor
|
|
{
|
|
fixed_point_scale_factor_binary,
|
|
fixed_point_scale_factor_decimal,
|
|
fixed_point_scale_factor_arbitrary
|
|
};
|
|
|
|
struct fixed_point_type_info
|
|
{
|
|
/* The scale factor is the value one has to multiply the actual data with
|
|
to get the fixed point value. We support three ways to encode it. */
|
|
enum fixed_point_scale_factor scale_factor_kind;
|
|
union
|
|
{
|
|
/* For a binary scale factor, the scale factor is 2 ** binary. */
|
|
int binary;
|
|
/* For a decimal scale factor, the scale factor is 10 ** decimal. */
|
|
int decimal;
|
|
/* For an arbitrary scale factor, the scale factor is the ratio
|
|
numerator / denominator. */
|
|
struct { tree numerator; tree denominator; } arbitrary;
|
|
} scale_factor;
|
|
};
|
|
|
|
void dwarf2out_c_finalize (void);
|
|
|
|
#endif /* GCC_DWARF2OUT_H */
|