GCC Middle and Back End API Reference
|
#include "expr.h"
Data Structures | |
struct | dw_cfi_row_struct |
struct | reg_saved_in_data_struct |
struct | dw_trace_info |
struct | trace_info_hasher |
struct | queued_reg_save |
Typedefs | |
typedef struct dw_cfi_row_struct | dw_cfi_row |
typedef struct reg_saved_in_data_struct | reg_saved_in_data |
typedef dw_trace_info * | dw_trace_info_ref |
Variables | |
static vec< dw_trace_info > | trace_info |
static vec< dw_trace_info_ref > | trace_work_list |
static hash_table < trace_info_hasher > | trace_index |
cfi_vec | cie_cfi_vec |
static dw_cfi_row * | cie_cfi_row |
static reg_saved_in_data * | cie_return_save |
static unsigned long | dwarf2out_cfi_label_num |
static rtx | add_cfi_insn |
static cfi_vec * | add_cfi_vec |
static dw_trace_info * | cur_trace |
static dw_cfi_row * | cur_row |
static dw_cfa_location * | cur_cfa |
static vec< queued_reg_save > | queued_reg_saves |
static bool | any_cfis_emitted |
static unsigned | dw_stack_pointer_regnum |
static unsigned | dw_frame_pointer_regnum |
static signed char | saved_do_cfi_asm = 0 |
typedef struct dw_cfi_row_struct dw_cfi_row |
A collected description of an entire row of the abstract CFI table.
typedef dw_trace_info* dw_trace_info_ref |
typedef struct reg_saved_in_data_struct reg_saved_in_data |
The caller's ORIG_REG is saved in SAVED_IN_REG.
|
static |
Add CFI either to the current insn stream or to a vector, or both.
References dw_cfi_row_struct::reg_save, vec_safe_grow_cleared(), and vec_safe_length().
Referenced by copy_cfi_row(), and dwarf2out_cfi_label().
|
static |
While we can occasionally have args_size < 0 internally, this state should not persist at a point we actually need an opcode.
|
static |
|
static |
Walk the function, looking for NOTE_INSN_CFI notes. Add the CFIs to the function's FDE, adding CFI labels and set_loc/advance_loc opcodes as necessary.
We always start with a function_begin label.
Don't attempt to advance_loc4 between labels in different sections.
Set the location counter to the new label.
|
static |
Return the insn before the first NOTE_INSN_CFI after START.
References dw_trace_info::head, dw_trace_info::id, memset(), and dw_trace_info::switch_sections.
bool cfa_equal_p | ( | ) |
Determine if two dw_cfa_location structures define the same data.
|
static |
Determine if two CFI entries are identical.
Make things easier for our callers, including missing operands.
Obviously, the opcodes must match.
Compare the two operands, re-using the type of the operands as already exposed elsewhere.
Referenced by cfi_oprnd_equal_p().
|
static |
Examine CFI and return true if a cfi label and set_loc is needed beforehand. Even when generating CFI assembler instructions, we still have to add the cfi to the list so that lookup_cfa_1 works later on. When -g2 and above we even need to force emitting of CFI labels and add to list a DW_CFA_set_loc for convert_cfa_to_fb_loc_list purposes. If we're generating DWARF3 output we use DW_OP_call_frame_cfa and so don't use convert_cfa_to_fb_loc_list.
References dw_cfi_row_struct::cfa, cur_row, cur_trace, dw_trace_info::end_true_args_size, HOST_WIDE_INT, maybe_record_trace_start(), cfa_loc::offset, and cfa_loc::reg.
|
static |
Determine if two CFI operands are identical.
References dw_cfi_row_struct::cfa_cfi, and cfi_equal_p().
|
static |
Determine if two CFI_ROW structures are identical.
Referenced by change_cfi_row().
|
static |
Emit CFI info to change the state from OLD_ROW to NEW_ROW.
References dw_trace_info::args_size_undefined, dw_trace_info::beg_row, dw_trace_info::beg_true_args_size, dw_trace_info::cfa_store, dw_trace_info::cfa_temp, cfi_row_equal_p(), copy_cfi_row(), cur_row, cur_trace, dump_file, dw_trace_info::end_true_args_size, get_trace_info(), HOST_WIDE_INT, dw_trace_info::id, dw_trace_info::regs_saved_in_regs, and rtx_name.
|
static |
Does INSN clobber any register which QUEUED_REG_SAVES lists a saved location for? Or, does it clobber a register which we've previously said that some other register is saved in, and for which we now have a new location for?
|
static |
Compare X and Y for equivalence. The inputs may be REGs or PC_RTX.
Referenced by notice_eh_throw(), and record_reg_saved_in_reg().
|
static |
Insert CFI notes between traces to properly change state between them.
??? Ideally, we should have both queued and processed every trace. However the current representation of constant pools on various targets is indistinguishable from unreachable code. Assume for the moment that we can simply skip over such traces.
??? Consider creating a DATA_INSN rtx code to indicate that these are not "real" instructions, and should not be considered. This could be generically useful for tablejump data as well.
Remove all unprocessed traces from the list.
Work from the end back to the beginning. This lets us easily insert remember/restore_state notes in the correct order wrt other notes.
In dwarf2out_switch_text_section, we'll begin a new FDE for the portion of the function in the alternate text section. The row state at the very beginning of that new FDE will be exactly the row state from the CIE.
If there's no change from the previous end state, fine.
Otherwise check for the common case of sharing state with the beginning of an epilogue, but not the end. Insert remember/restore opcodes in that case.
Note that if we blindly insert the remember at the start of the trace, we can wind up increasing the size of the unwind info due to extra advance opcodes. Instead, put the remember immediately before the next state change. We know there must be one, because the state at the beginning and head of the trace differ.
Otherwise, we'll simply change state from the previous end.
Connect args_size between traces that have can_throw_internal insns.
??? Search back to previous CFI note.
References dump_file, hash_table< Descriptor, Allocator >::find_slot_with_hash(), dw_trace_info::head, rtx_name, and dw_trace_info::switch_sections.
|
static |
Return a copy of an existing CFI row.
References add_cfi(), dw_cfi_oprnd_struct::dw_cfi_offset, dw_cfi_struct::dw_cfi_opc, dw_cfi_struct::dw_cfi_oprnd1, and new_cfi().
Referenced by change_cfi_row().
|
static |
Scan the function and create the initial set of CFI notes.
Always begin at the entry trace.
References save_point_p().
|
static |
On entry, the Canonical Frame Address is at SP.
For a few targets, we have the return address incoming into a register, but choose a different return column. This will result in a DW_CFA_register for the return, and an entry in regs_saved_in_regs to match. If the target later stores that return address register to the stack, we want to be able to emit the DW_CFA_offset against the return column, not the intermediate save register. Save the contents of regs_saved_in_regs so that we can re-initialize it at the start of each function.
|
static |
Set up the pseudo-cfg of instruction traces, as described at the block comment at the top of the file.
The first trace begins at the start of the function, and begins with the CIE row state.
Walk all the insns, collecting start of trace locations.
We should have just seen a barrier.
Watch out for save_point notes between basic blocks. In particular, a note after a barrier. Do not record these, delaying trace creation until the label.
Create the trace index after we've finished building trace_info, avoiding stale pointer problems due to reallocation.
|
static |
Propagate CUR_TRACE state to the destinations implied by INSN.
??? Sadly, this is in large part a duplicate of make_edges.
Sibling calls don't have edges inside this function.
Process non-local goto edges.
Process EH edges.
References add_cfi_insn, dw_cfi_row_struct::cfa, cur_cfa, cur_row, cur_trace, dw_trace_info::end_true_args_size, HOST_WIDE_INT, dw_cfi_row_struct::reg_save, scan_insn_after(), and vec_safe_copy().
void debug_cfi_row | ( | dw_cfi_row * | row | ) |
void debug_cfi_row | ( | ) |
|
static |
The CFA is now calculated from NEW_CFA. Consider OLD_CFA in determining what opcode to emit. Returns the CFI opcode to effect the change, or NULL if NEW_CFA == OLD_CFA.
If nothing changed, no need to issue any call frame instructions.
Construct a "DW_CFA_def_cfa_offset <offset>" instruction, indicating the CFA register did not change but the offset did. The data factoring for DW_CFA_def_cfa_offset_sf happens in output_cfi, or in the assembler via the .cfi_def_cfa_offset directive.
Construct a "DW_CFA_def_cfa_register <register>" instruction, indicating the CFA register has changed to <register> but the offset has not changed.
Construct a "DW_CFA_def_cfa <register> <offset>" instruction, indicating the CFA register has changed to <register> with the specified offset. The data factoring for DW_CFA_def_cfa_sf happens in output_cfi, or in the assembler via the .cfi_def_cfa directive.
Construct a DW_CFA_def_cfa_expression instruction to calculate the CFA using a full location expression since no register-offset pair is available.
|
static |
Similarly, but take OLD_CFA from CUR_ROW, and update it after the fact.
|
inlinestatic |
Divide OFF by DWARF_CIE_DATA_ALIGNMENT, asserting no remainder.
|
static |
|
static |
Generate a new label for the CFI info to refer to.
References add_cfi(), dw_cfi_struct::dw_cfi_opc, dw_cfi_struct::dw_cfi_oprnd1, dw_cfi_oprnd_struct::dw_cfi_reg_num, and new_cfi().
bool dwarf2out_do_cfi_asm | ( | void | ) |
Decide whether to emit frame unwind via assembler directives.
Assume failure for a moment.
Make sure the personality encoding is one the assembler can support. In particular, aligned addresses can't be handled.
If we can't get the assembler to emit only .debug_frame, and we don't need dwarf2 unwind info for exceptions, then emit .debug_frame by hand.
Success!
Referenced by append_entry_to_tmpl_value_parm_die_table().
bool dwarf2out_do_frame | ( | void | ) |
Decide whether we want to emit frame unwind information for the current translation unit.
We want to emit correct CFA location expressions or lists, so we have to return true if we're going to output debug info, even if we're not going to output frame or unwind info.
void dwarf2out_emit_cfi | ( | ) |
|
static |
Output all the entries in QUEUED_REG_SAVES.
References cfa_loc::base_offset, cur_cfa, dwf_regno(), cfa_loc::indirect, memset(), cfa_loc::offset, and cfa_loc::reg.
|
static |
Record call frame debugging information for INSN, which either sets SP or FP (adjusting how we calculate the frame address) or saves a register to the stack.
The actual flush happens elsewhere.
Check again. A parallel can save and update the same register. We could probably check just once, here, but this is safer than removing the check at the start of the function.
|
static |
A subroutine of dwarf2out_frame_debug, process a REG_ADJUST_CFA note.
References dwf_regno(), and reg_save().
|
static |
A subroutine of dwarf2out_frame_debug, process a REG_CFA_EXPRESSION note.
??? We'd like to use queue_reg_save, were the interface different, and, as above, we could manage flushing for epilogues.
|
static |
A subroutine of dwarf2out_frame_debug, process a REG_CFA_OFFSET note.
As documented, only consider extremely simple addresses.
??? We'd like to use queue_reg_save, but we need to come up with a different flushing heuristic for epilogues.
We have a PARALLEL describing where the contents of SRC live. Queue register saves for each piece of the PARALLEL.
|
static |
A subroutine of dwarf2out_frame_debug, process a REG_CFA_REGISTER note.
??? We'd like to use queue_reg_save, but we need to come up with a different flushing heuristic for epilogues.
|
static |
A subroutine of dwarf2out_frame_debug, process a REG_CFA_RESTORE note.
|
static |
A subroutine of dwarf2out_frame_debug, process a REG_CFA_WINDOW_SAVE. ??? Perhaps we should note in the CIE where windows are saved (instead of assuming 0(cfa)) and what registers are in the window.
References dw_trace_info::cfa_temp, cfun, cur_cfa, cur_trace, dwarf2out_frame_debug_expr(), dwf_regno(), function::fde, HOST_WIDE_INT, limit, cfa_loc::offset, offset, cfa_loc::reg, reg_saved_in(), SET, and dw_fde_struct::stack_realign.
|
static |
A subroutine of dwarf2out_frame_debug, process a REG_DEF_CFA note.
??? If this fails, we could be calling into the _loc functions to define a full expression. So far no port does that.
|
static |
Record call frame debugging information for an expression EXPR, which either sets SP or FP (adjusting how we calculate the frame address) or saves a register to the stack or another register. LABEL indicates the address of EXPR. This function encodes a state machine mapping rtxes to actions on cfa, cfa_store, and cfa_temp.reg. We describe these rules so users need not read the source code. The High-Level Picture Changes in the register we use to calculate the CFA: Currently we assume that if you copy the CFA register into another register, we should take the other one as the new CFA register; this seems to work pretty well. If it's wrong for some target, it's simple enough not to set RTX_FRAME_RELATED_P on the insn in question. Changes in the register we use for saving registers to the stack: This is usually SP, but not always. Again, we deduce that if you copy SP into another register (and SP is not the CFA register), then the new register is the one we will be using for register saves. This also seems to work. Register saves: There's not much guesswork about this one; if RTX_FRAME_RELATED_P is set on an insn which modifies memory, it's a register save, and the register used to calculate the destination had better be the one we think we're using for this purpose. It's also assumed that a copy from a call-saved register to another register is saving that register if RTX_FRAME_RELATED_P is set on that instruction. If the copy is from a call-saved register to the *same* register, that means that the register is now the same value as in the caller. Except: If the register being saved is the CFA register, and the offset is nonzero, we are saving the CFA, so we assume we have to use DW_CFA_def_cfa_expression. If the offset is 0, we assume that the intent is to save the value of SP from the previous frame. In addition, if a register has previously been saved to a different register, Invariants / Summaries of Rules cfa current rule for calculating the CFA. It usually consists of a register and an offset. This is actually stored in *cur_cfa, but abbreviated for the purposes of this documentation. cfa_store register used by prologue code to save things to the stack cfa_store.offset is the offset from the value of cfa_store.reg to the actual CFA cfa_temp register holding an integral value. cfa_temp.offset stores the value, which will be used to adjust the stack pointer. cfa_temp is also used like cfa_store, to track stores to the stack via fp or a temp reg. Rules 1- 4: Setting a register's value to cfa.reg or an expression with cfa.reg as the first operand changes the cfa.reg and its cfa.offset. Rule 1 and 4 also set cfa_temp.reg and cfa_temp.offset. Rules 6- 9: Set a non-cfa.reg register value to a constant or an expression yielding a constant. This sets cfa_temp.reg and cfa_temp.offset. Rule 5: Create a new register cfa_store used to save items to the stack. Rules 10-14: Save a register to the stack. Define offset as the difference of the original location and cfa_store's location (or cfa_temp's location if cfa_temp is used). Rules 16-20: If AND operation happens on sp in prologue, we assume stack is realigned. We will use a group of DW_OP_XXX expressions to represent the location of the stored register instead of CFA+offset. The Rules "{a,b}" indicates a choice of a xor b. "<reg>:cfa.reg" indicates that <reg> must equal cfa.reg. Rule 1: (set <reg1> <reg2>:cfa.reg) effects: cfa.reg = <reg1> cfa.offset unchanged cfa_temp.reg = <reg1> cfa_temp.offset = cfa.offset Rule 2: (set sp ({minus,plus,losum} {sp,fp}:cfa.reg {<const_int>,<reg>:cfa_temp.reg})) effects: cfa.reg = sp if fp used cfa.offset += {+/- <const_int>, cfa_temp.offset} if cfa.reg==sp cfa_store.offset += {+/- <const_int>, cfa_temp.offset} if cfa_store.reg==sp Rule 3: (set fp ({minus,plus,losum} <reg>:cfa.reg <const_int>)) effects: cfa.reg = fp cfa_offset += +/- <const_int> Rule 4: (set <reg1> ({plus,losum} <reg2>:cfa.reg <const_int>)) constraints: <reg1> != fp <reg1> != sp effects: cfa.reg = <reg1> cfa_temp.reg = <reg1> cfa_temp.offset = cfa.offset Rule 5: (set <reg1> (plus <reg2>:cfa_temp.reg sp:cfa.reg)) constraints: <reg1> != fp <reg1> != sp effects: cfa_store.reg = <reg1> cfa_store.offset = cfa.offset - cfa_temp.offset Rule 6: (set <reg> <const_int>) effects: cfa_temp.reg = <reg> cfa_temp.offset = <const_int> Rule 7: (set <reg1>:cfa_temp.reg (ior <reg2>:cfa_temp.reg <const_int>)) effects: cfa_temp.reg = <reg1> cfa_temp.offset |= <const_int> Rule 8: (set <reg> (high <exp>)) effects: none Rule 9: (set <reg> (lo_sum <exp> <const_int>)) effects: cfa_temp.reg = <reg> cfa_temp.offset = <const_int> Rule 10: (set (mem ({pre,post}_modify sp:cfa_store (???? <reg1> <const_int>))) <reg2>) effects: cfa_store.offset -= <const_int> cfa.offset = cfa_store.offset if cfa.reg == sp cfa.reg = sp cfa.base_offset = -cfa_store.offset Rule 11: (set (mem ({pre_inc,pre_dec,post_dec} sp:cfa_store.reg)) <reg>) effects: cfa_store.offset += -/+ mode_size(mem) cfa.offset = cfa_store.offset if cfa.reg == sp cfa.reg = sp cfa.base_offset = -cfa_store.offset Rule 12: (set (mem ({minus,plus,losum} <reg1>:{cfa_store,cfa_temp} <const_int>)) <reg2>) effects: cfa.reg = <reg1> cfa.base_offset = -/+ <const_int> - {cfa_store,cfa_temp}.offset Rule 13: (set (mem <reg1>:{cfa_store,cfa_temp}) <reg2>) effects: cfa.reg = <reg1> cfa.base_offset = -{cfa_store,cfa_temp}.offset Rule 14: (set (mem (post_inc <reg1>:cfa_temp <const_int>)) <reg2>) effects: cfa.reg = <reg1> cfa.base_offset = -cfa_temp.offset cfa_temp.offset -= mode_size(mem) Rule 15: (set <reg> {unspec, unspec_volatile}) effects: target-dependent Rule 16: (set sp (and: sp <const_int>)) constraints: cfa_store.reg == sp effects: cfun->fde.stack_realign = 1 cfa_store.offset = 0 fde->drap_reg = cfa.reg if cfa.reg != sp and cfa.reg != fp Rule 17: (set (mem ({pre_inc, pre_dec} sp)) (mem (plus (cfa.reg) (const_int)))) effects: cfa_store.offset += -/+ mode_size(mem) Rule 18: (set (mem ({pre_inc, pre_dec} sp)) fp) constraints: fde->stack_realign == 1 effects: cfa_store.offset = 0 cfa.reg != HARD_FRAME_POINTER_REGNUM Rule 19: (set (mem ({pre_inc, pre_dec} sp)) cfa.reg) constraints: fde->stack_realign == 1 && cfa.offset == 0 && cfa.indirect == 0 && cfa.reg != HARD_FRAME_POINTER_REGNUM effects: Use DW_CFA_def_cfa_expression to define cfa cfa.reg == fde->drap_reg
If RTX_FRAME_RELATED_P is set on a PARALLEL, process each member of the PARALLEL independently. The first element is always processed if it is a SET. This is for backward compatibility. Other elements are processed only if they are SETs and the RTX_FRAME_RELATED_P flag is set in them.
PARALLELs have strict read-modify-write semantics, so we ought to evaluate every rvalue before changing any lvalue. It's cumbersome to do that in general, but there's an easy approximation that is enough for all current users: handle register saves before register assignments.
Setting FP from SP.
Rule 1
Update the CFA rule wrt SP or FP. Make sure src is relative to the current CFA register. We used to require that dest be either SP or FP, but the ARM copies SP to a temporary register, and from there to FP. So we just rely on the backends to only set RTX_FRAME_RELATED_P on appropriate insns.
Saving a register in a register.
For the SPARC and its register window.
After stack is aligned, we can only save SP in FP if drap register is used. In this case, we have to restore stack pointer with the CFA value and we don't generate this DWARF information.
Rule 2
Adjusting SP.
Restoring SP from FP in the epilogue.
Assume we've set the source reg of the LO_SUM from sp.
Rule 3
Either setting the FP from an offset of the SP, or adjusting the FP
Rule 4
Setting a temporary CFA register that will be copied into the FP later on.
Or used to save regs to the stack.
Rule 5
Setting a scratch register that we will use instead of SP for saving registers to the stack.
Rule 9
Rule 6
Rule 7
Skip over HIGH, assuming it will be followed by a LO_SUM, which will fill in all of the bits.
Rule 8
Rule 15
All unspecs should be represented by REG_CFA_* notes.
Rule 16
If this AND operation happens on stack pointer in prologue, we assume the stack is realigned and we extract the alignment.
We interpret reg_save differently with stack_realign set. Thus we must flush whatever we have queued first.
Saving a register to the stack. Make sure dest is relative to the CFA register.
Rule 10
With a push.
We can't handle variable size modifications.
Rule 11
Rule 18: If stack is aligned, we will use FP as a reference to represent the address of the stored regiser.
Rule 12
With an offset.
Rule 13
Without an offset.
Rule 14
Rule 17
If the source operand of this MEM operation is a memory, we only care how much stack grew.
We're storing the current CFA reg into the stack.
Rule 19
If stack is aligned, putting CFA reg into stack means we can no longer use reg + offset to represent CFA. Here we use DW_CFA_def_cfa_expression instead. The result of this expression equals to the original CFA value.
If the source register is exactly the CFA, assume we're saving SP like any other register; this happens on the ARM.
Otherwise, we'll need to look in the stack to calculate the CFA.
We have a PARALLEL describing where the contents of SRC live. Queue register saves for each piece of the PARALLEL.
Referenced by dwarf2out_frame_debug_cfa_window_save().
|
static |
Convert a DWARF call frame info. operation to its string name
References asm_out_file.
|
inlinestatic |
Short-hand inline for the very common D_F_R (REGNO (x)) operation.
??? This ought to go into dwarf2out.h, except that dwarf2out.h is used in places where rtl is prohibited.
References pc_rtx, record_reg_saved_in_reg(), queued_reg_save::reg, and queued_reg_save::saved_reg.
Referenced by dwarf2out_flush_queued_reg_saves(), dwarf2out_frame_debug_adjust_cfa(), dwarf2out_frame_debug_cfa_window_save(), and reg_saved_in().
|
static |
Annotate the function with NOTE_INSN_CFI notes to record the CFI state at each location within the function. These notes will be emitted during pass_final.
The first time we're called, compute the incoming frame state.
Do the work.
Free all the data we allocated.
rtx expand_builtin_dwarf_sp_column | ( | void | ) |
Hook used by __throw.
References choose_hard_reg_mode(), and HOST_WIDE_INT.
void expand_builtin_init_dwarf_reg_sizes | ( | ) |
Generate code to initialize the register size table.
References targetm.
|
static |
Targets which still implement the prologue in assembler text cannot use the generic dwarf2 unwinding.
??? What to do for UI_TARGET unwinding? They might be able to benefit from the optimized shrink-wrapping annotations that we will compute. For now, only produce the CFI notes for dwarf2.
|
static |
This function fills in aa dw_cfa_location structure from a dwarf location descriptor sequence.
|
static |
Referenced by change_cfi_row().
|
static |
MEM is a memory reference for the register size table, each element of which has mode MODE. Initialize column C as a return address column.
|
static |
Record the initial position of the return address. RTL is INCOMING_RETURN_ADDR_RTX.
RA is in a register.
RA is on the stack.
The return address is at some offset from any value we can actually load. For instance, on the SPARC it is in %i7+8. Just ignore the offset for now; it doesn't matter for unwinding frames.
void lookup_cfa_1 | ( | ) |
Find the previous value for the CFA, iteratively. CFI is the opcode to interpret, *LOC will be updated as necessary, *REMEMBER is used for one level of remember/restore state processing.
References dw_cfi_oprnd_struct::dw_cfi_addr, dw_cfi_oprnd_struct::dw_cfi_loc, dw_cfi_oprnd_struct::dw_cfi_offset, dw_cfi_oprnd_addr, dw_cfi_oprnd_loc, dw_cfi_oprnd_offset, dw_cfi_oprnd_reg_num, dw_cfi_oprnd_unused, dw_cfi_oprnd_struct::dw_cfi_reg_num, and loc_descr_equal_p().
rtl_opt_pass* make_pass_dwarf2_frame | ( | ) |
|
static |
If LABEL is the start of a trace, then initialize the state of that trace from CUR_TRACE and CUR_ROW.
This is the first time we've encountered this trace. Propagate state across the edge and push the trace onto the work list.
We ought to have the same state incoming to a given trace no matter how we arrive at the trace. Anything else means we've got some kind of optimization error.
The args_size is allowed to conflict if it isn't actually used.
Referenced by cfi_label_required_p().
|
static |
Similarly, but handle the args_size and CFA reset across EH and non-local goto edges.
Convert a change in args_size (always a positive in the direction of stack growth) to a change in stack pointer.
|
inlinestatic |
Return true if we need a signed version of a given opcode (e.g. DW_CFA_offset_extended_sf vs DW_CFA_offset_extended).
|
inlinestatic |
Return a pointer to a newly allocated Call Frame Instruction.
Referenced by copy_cfi_row(), and dwarf2out_cfi_label().
|
static |
Return a newly allocated CFI row, with no defined data.
References add_cfi_insn, and emit_note_after().
|
static |
A subroutine of scan_trace. Check INSN for a REG_ARGS_SIZE note and adjust data structures to match.
If the CFA is computed off the stack pointer, then we must adjust the computation of the CFA as well.
Convert a change in args_size (always a positive in the direction of stack growth) to a change in stack pointer.
|
static |
A subroutine of scan_trace. INSN is can_throw_internal. Update the data within the trace related to EH insns and args_size.
??? If the CFA is the stack pointer, search backward for the last CFI note and insert there. Given that the stack changed for the args_size change, there *must* be such a note in between here and the last eh insn.
References compare_reg_or_pc(), and queued_reg_save::reg.
|
static |
This routine will generate the correct assembly data for a location description based on a cfi entry with a complex address.
Output the size of the block.
Now output the operations themselves.
|
static |
Similar, but used for .cfi_escape.
Output the size of the block.
Now output the operations themselves.
void output_cfi | ( | ) |
Output a Call Frame Information opcode and its operand(s).
Obsoleted by DW_CFA_offset_extended_sf.
References dw2_asm_output_data_uleb128_raw(), dw_cfi_oprnd_struct::dw_cfi_offset, dw_cfi_struct::dw_cfi_oprnd1, and HOST_WIDE_INT_PRINT_DEC.
void output_cfi_directive | ( | ) |
Similar, but do it via assembler directives instead.
Should only be created in a code path not followed when emitting via directives. The assembler is going to take care of this for us. But this routines is also used for debugging dumps, so print something.
FALLTHRU
|
static |
Add an entry to QUEUED_REG_SAVES saying that REG is now saved at SREG, or if SREG is NULL then it is saved at OFFSET to the CFA.
Duplicates waste space, but it's also necessary to remove them for correctness, since the queue gets output in reverse order.
References cur_trace, reg_saved_in_data_struct::orig_reg, queued_reg_save::reg, dw_trace_info::regs_saved_in_regs, reg_saved_in_data_struct::saved_in_reg, and queued_reg_save::saved_reg.
|
static |
Record SRC as being saved in DEST. DEST may be null to delete an existing entry. SRC may be a register or PC_RTX.
References compare_reg_or_pc(), cur_trace, modified_in_p(), reg_saved_in_data_struct::orig_reg, queued_reg_save::reg, dw_trace_info::regs_saved_in_regs, and reg_saved_in_data_struct::saved_in_reg.
Referenced by dwf_regno().
|
static |
Add the CFI for saving a register. REG is the CFA column number. If SREG is -1, the register is saved at OFFSET from the CFA; otherwise it is saved in SREG.
When stack is aligned, store REG using DW_CFA_expression with FP.
While we could emit something like DW_CFA_same_value or DW_CFA_restore, we never expect to see something like that in a prologue. This is more likely to be a bug. A backend can always bypass this by using REG_CFA_RESTORE directly.
References cur_cfa, cfa_loc::indirect, and cfa_loc::offset.
Referenced by dwarf2out_frame_debug_adjust_cfa().
|
static |
What register, if any, is currently saved in REG?
References cur_cfa, dwf_regno(), HOST_WIDE_INT, cfa_loc::offset, offset, and cfa_loc::reg.
Referenced by dwarf2out_frame_debug_cfa_window_save().
|
static |
Labels, except those that are really jump tables.
We split traces at the prologue/epilogue notes because those are points at which the unwind info is usually stable. This makes it easier to find spots with identical unwind info so that we can use remember/restore_state opcodes.
Referenced by create_cfi_notes().
|
static |
A subroutine of scan_trace. Do what needs to be done "after" INSN.
Referenced by create_trace_edges().
|
static |
Scan the trace beginning at INSN and create the CFI notes for the instructions therein.
Do everything that happens "before" the insn.
Notice the end of a trace.
Don't bother saving the unneeded queued registers at all.
Propagate across fallthru edges.
Handle all changes to the row state. Sequences require special handling for the positioning of the notes.
??? Hopefully multiple delay slots are not annulled.
If ELT is an instruction from target of an annulled branch, the effects are for the target only and so the args_size and CFA along the current path shouldn't change.
??? Should we instead save the entire row state?
If ELT is a annulled branch-taken instruction (i.e. executed only when branch is not taken), the args_size and CFA should not change through the jump.
Update and continue with the trace.
The insns in the delay slot should all be considered to happen "before" a call insn. Consider a call with a stack pointer adjustment in the delay slot. The backtrace from the callee should include the sp adjustment. Unfortunately, that leaves us with an unavoidable unwinding error exactly at the call insn itself. For jump insns we'd prefer to avoid this error by placing the notes after the sequence.
Make sure any register saves are visible at the jump target.
However, if there is some adjustment on the call itself, e.g. a call_pop, that action should be considered to happen after the call returns.
Flush data before calls and jumps, and of course if necessary.
Between frame-related-p and args_size we might have otherwise emitted two cfa adjustments. Do it now.
Minimize the number of advances by emitting the entire queue once anything is emitted.
Note that a test for control_flow_insn_p does exactly the same tests as are done to actually create the edges. So always call the routine and let it not create edges for non-control-flow insns.
|
static |
Perform ROW->REG_SAVE[COLUMN] = CFI. CFI may be null, indicating that the register column is no longer saved.
|
static |
The insn after which a new CFI note should be emitted.
Referenced by create_trace_edges(), and new_cfi_row().
|
static |
When non-null, add_cfi will add the CFI to this vector.
|
static |
True if any CFI directives were emitted at the current insn.
|
static |
The state of the first row of the FDE table, which includes the state provided by the CIE.
cfi_vec cie_cfi_vec |
A vector of call frame insns for the CIE.
|
static |
|
static |
A copy of the current CFA, for use during the processing of a single insn.
Referenced by create_trace_edges(), dwarf2out_flush_queued_reg_saves(), dwarf2out_frame_debug_cfa_window_save(), reg_save(), and reg_saved_in().
|
static |
The current, i.e. most recently generated, row of the CFI table.
Referenced by cfi_label_required_p(), change_cfi_row(), and create_trace_edges().
|
static |
The current instruction trace.
Referenced by cfi_label_required_p(), change_cfi_row(), create_trace_edges(), dwarf2out_frame_debug_cfa_window_save(), queue_reg_save(), and record_reg_saved_in_reg().
|
static |
|
static |
Short-hand for commonly used register numbers.
|
static |
|
static |
|
static |
Save the result of dwarf2out_do_frame across PCH. This variable is tri-state, with 0 unset, >0 true, <0 false.
|
static |
|
static |
The variables making up the pseudo-cfg, as described above.
|
static |