aboutsummaryrefslogtreecommitdiff
path: root/gcc/frame-dwarf2.c
diff options
context:
space:
mode:
authorAndrew Macleod <amacleod@cygnus.com>2000-06-26 16:15:07 -0400
committerJason Merrill <jason@gcc.gnu.org>2000-06-26 16:15:07 -0400
commit7d9d8943c9a920f25b1b0319dc201de1a280c412 (patch)
tree71b921aa27ff78a41f60dae65d13fa8833d71768 /gcc/frame-dwarf2.c
parent4540727afe11267220f725c694e04e01de19f9d0 (diff)
downloadgcc-7d9d8943c9a920f25b1b0319dc201de1a280c412.zip
gcc-7d9d8943c9a920f25b1b0319dc201de1a280c412.tar.gz
gcc-7d9d8943c9a920f25b1b0319dc201de1a280c412.tar.bz2
dwarf2.h (enum dwarf_call_frame_info): Add DW_CFA_def_cfa_expression.
* dwarf2.h (enum dwarf_call_frame_info): Add DW_CFA_def_cfa_expression. * dwarf2out.c (union dw_cfi_oprnd_struct): Add a pointer to a dw_loc_descr_struct entry. (struct cfa_loc): New structure to track a CFA location. (lookup_cfa): Take a cfa_loc parameter instead of a reg and an offset. (lookup_cfa_1): Take a cfa_loc parameter instead of a reg and an offset, plus handle DW_CFA_def_cfa_expression. (def_cfa_1): Use to be dwarf2out_def_cfa, only now it uses a cfa_loc record. (dwarf2out_def_cfa): Entry point maintained for compatability. (dwarf_cfi_name): Add DW_CFA_def_cfa_expression. (cfa_reg, cfa_offset): Replace with cfa_loc record 'cfa'. (cfa_store_reg, cfa_store_offset): Replace with cfa_loc 'cfa_store'. (initial_return_save, dwarf2out_stack_adjust): Use cfa.reg, not cfa_reg. (dwarf2out_frame_debug_expr): Use new cfa_loc records. Recognize rtl sequences for the new DW_CFA_def_cfa_expression record. (dwarf2out_frame_debug): Use new variables/fields.A (output_cfi): Handle DW_CFA_def_cfa_expression. (output_cfa_loc): New function to generate a CFI record for DW_CFA_def_cfa_expression. (get_cfa_from_loc_descr): New function to get a cfa_loc record from a dw_loc_descr sequeunce. (build_loc_descr): Build a dw_loc_descr from a cfa_loc record. (dwarf_stack_op_name, new_loc_descr, add_loc_descr, size_of_loc_descr, size_of_locs, output_loc_operands, output_loc_sequence): Move into unwind info section. * frame.h (frame_state): Add base_offset and indirect fields. * frame-dwarf2.c (decode_stack_op): New function to interpret a dw_loc_descr operation. (execute_cfa_insn): Add support for DW_CFA_def_cfa_expression. (struct frame_state): Add base offset and indirect fields. * libgcc2.c (next_stack_level): Support indirect loading for CFA. From-SVN: r34717
Diffstat (limited to 'gcc/frame-dwarf2.c')
-rw-r--r--gcc/frame-dwarf2.c116
1 files changed, 116 insertions, 0 deletions
diff --git a/gcc/frame-dwarf2.c b/gcc/frame-dwarf2.c
index fb2e9fb..52db16f5 100644
--- a/gcc/frame-dwarf2.c
+++ b/gcc/frame-dwarf2.c
@@ -466,6 +466,108 @@ extract_cie_info (fde *f, struct cie_info *c)
return p;
}
+/* Decode a DW_OP stack operation. */
+
+static void *
+decode_stack_op (unsigned char *buf, struct frame_state *state)
+{
+ enum dwarf_location_atom op;
+ int offset;
+
+ op = *buf++;
+ switch (op)
+ {
+ case DW_OP_reg0:
+ case DW_OP_reg1:
+ case DW_OP_reg2:
+ case DW_OP_reg3:
+ case DW_OP_reg4:
+ case DW_OP_reg5:
+ case DW_OP_reg6:
+ case DW_OP_reg7:
+ case DW_OP_reg8:
+ case DW_OP_reg9:
+ case DW_OP_reg10:
+ case DW_OP_reg11:
+ case DW_OP_reg12:
+ case DW_OP_reg13:
+ case DW_OP_reg14:
+ case DW_OP_reg15:
+ case DW_OP_reg16:
+ case DW_OP_reg17:
+ case DW_OP_reg18:
+ case DW_OP_reg19:
+ case DW_OP_reg20:
+ case DW_OP_reg21:
+ case DW_OP_reg22:
+ case DW_OP_reg23:
+ case DW_OP_reg24:
+ case DW_OP_reg25:
+ case DW_OP_reg26:
+ case DW_OP_reg27:
+ case DW_OP_reg28:
+ case DW_OP_reg29:
+ case DW_OP_reg30:
+ case DW_OP_reg31:
+ state->cfa_reg = op - DW_OP_reg0;
+ break;
+ case DW_OP_regx:
+ buf = decode_sleb128 (buf, &offset);
+ state->cfa_reg = offset;
+ case DW_OP_breg0:
+ case DW_OP_breg1:
+ case DW_OP_breg2:
+ case DW_OP_breg3:
+ case DW_OP_breg4:
+ case DW_OP_breg5:
+ case DW_OP_breg6:
+ case DW_OP_breg7:
+ case DW_OP_breg8:
+ case DW_OP_breg9:
+ case DW_OP_breg10:
+ case DW_OP_breg11:
+ case DW_OP_breg12:
+ case DW_OP_breg13:
+ case DW_OP_breg14:
+ case DW_OP_breg15:
+ case DW_OP_breg16:
+ case DW_OP_breg17:
+ case DW_OP_breg18:
+ case DW_OP_breg19:
+ case DW_OP_breg20:
+ case DW_OP_breg21:
+ case DW_OP_breg22:
+ case DW_OP_breg23:
+ case DW_OP_breg24:
+ case DW_OP_breg25:
+ case DW_OP_breg26:
+ case DW_OP_breg27:
+ case DW_OP_breg28:
+ case DW_OP_breg29:
+ case DW_OP_breg30:
+ case DW_OP_breg31:
+ state->cfa_reg = op - DW_OP_breg0;
+ buf = decode_sleb128 (buf, &offset);
+ state->base_offset = offset;
+ break;
+ case DW_OP_bregx:
+ buf = decode_sleb128 (buf, &offset);
+ state->cfa_reg = offset;
+ buf = decode_sleb128 (buf, &offset);
+ state->base_offset = offset;
+ break;
+ case DW_OP_deref:
+ state->indirect = 1;
+ break;
+ case DW_OP_plus_uconst:
+ buf = decode_uleb128 (buf, &offset);
+ state->cfa_offset = offset;
+ break;
+ default:
+ abort ();
+ }
+ return buf;
+}
/* Decode one instruction's worth of DWARF 2 call frame information.
Used by __frame_state_for. Takes pointers P to the instruction to
decode, STATE to the current register unwind information, INFO to the
@@ -567,6 +669,20 @@ execute_cfa_insn (void *p, struct frame_state_internal *state,
p = decode_uleb128 (p, &offset);
state->s.cfa_offset = offset;
break;
+ case DW_CFA_def_cfa_expression:
+ {
+ void *end;
+ state->s.cfa_reg = 0;
+ state->s.cfa_offset = 0;
+ state->s.base_offset = 0;
+ state->s.indirect = 0;
+
+ p = decode_uleb128 (p, &offset);
+ end = p + offset;
+ while (p < end)
+ p = decode_stack_op (p, &(state->s));
+ break;
+ }
case DW_CFA_remember_state:
{