1 /* Implements exception handling.
2 Copyright (C) 1989, 1992, 1993, 1994, 1995, 1996, 1997, 1998,
3 1999, 2000, 2001, 2002, 2003, 2004 Free Software Foundation, Inc.
4 Contributed by Mike Stump <mrs@cygnus.com>.
6 This file is part of GCC.
8 GCC is free software; you can redistribute it and/or modify it under
9 the terms of the GNU General Public License as published by the Free
10 Software Foundation; either version 2, or (at your option) any later
13 GCC is distributed in the hope that it will be useful, but WITHOUT ANY
14 WARRANTY; without even the implied warranty of MERCHANTABILITY or
15 FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
18 You should have received a copy of the GNU General Public License
19 along with GCC; see the file COPYING. If not, write to the Free
20 Software Foundation, 59 Temple Place - Suite 330, Boston, MA
24 /* An exception is an event that can be signaled from within a
25 function. This event can then be "caught" or "trapped" by the
26 callers of this function. This potentially allows program flow to
27 be transferred to any arbitrary code associated with a function call
28 several levels up the stack.
30 The intended use for this mechanism is for signaling "exceptional
31 events" in an out-of-band fashion, hence its name. The C++ language
32 (and many other OO-styled or functional languages) practically
33 requires such a mechanism, as otherwise it becomes very difficult
34 or even impossible to signal failure conditions in complex
35 situations. The traditional C++ example is when an error occurs in
36 the process of constructing an object; without such a mechanism, it
37 is impossible to signal that the error occurs without adding global
38 state variables and error checks around every object construction.
40 The act of causing this event to occur is referred to as "throwing
41 an exception". (Alternate terms include "raising an exception" or
42 "signaling an exception".) The term "throw" is used because control
43 is returned to the callers of the function that is signaling the
44 exception, and thus there is the concept of "throwing" the
45 exception up the call stack.
47 [ Add updated documentation on how to use this. ] */
52 #include "coretypes.h"
60 #include "insn-config.h"
62 #include "integrate.h"
63 #include "hard-reg-set.h"
64 #include "basic-block.h"
66 #include "dwarf2asm.h"
67 #include "dwarf2out.h"
75 #include "langhooks.h"
78 /* Provide defaults for stuff that may not be defined when using
80 #ifndef EH_RETURN_DATA_REGNO
81 #define EH_RETURN_DATA_REGNO(N) INVALID_REGNUM
85 /* Protect cleanup actions with must-not-throw regions, with a call
86 to the given failure handler. */
87 tree (*lang_protect_cleanup_actions
) (void);
89 /* Return true if type A catches type B. */
90 int (*lang_eh_type_covers
) (tree a
, tree b
);
92 /* Map a type to a runtime object to match type. */
93 tree (*lang_eh_runtime_type
) (tree
);
95 /* A hash table of label to region number. */
97 struct ehl_map_entry
GTY(())
100 struct eh_region
*region
;
103 static GTY(()) int call_site_base
;
104 static GTY ((param_is (union tree_node
)))
105 htab_t type_to_runtime_map
;
107 /* Describe the SjLj_Function_Context structure. */
108 static GTY(()) tree sjlj_fc_type_node
;
109 static int sjlj_fc_call_site_ofs
;
110 static int sjlj_fc_data_ofs
;
111 static int sjlj_fc_personality_ofs
;
112 static int sjlj_fc_lsda_ofs
;
113 static int sjlj_fc_jbuf_ofs
;
115 /* Describes one exception region. */
116 struct eh_region
GTY(())
118 /* The immediately surrounding region. */
119 struct eh_region
*outer
;
121 /* The list of immediately contained regions. */
122 struct eh_region
*inner
;
123 struct eh_region
*next_peer
;
125 /* An identifier for this region. */
128 /* When a region is deleted, its parents inherit the REG_EH_REGION
129 numbers already assigned. */
132 /* Each region does exactly one thing. */
139 ERT_ALLOWED_EXCEPTIONS
,
145 /* Holds the action to perform based on the preceding type. */
147 /* A list of catch blocks, a surrounding try block,
148 and the label for continuing after a catch. */
149 struct eh_region_u_try
{
150 struct eh_region
*catch;
151 struct eh_region
*last_catch
;
152 struct eh_region
*prev_try
;
154 } GTY ((tag ("ERT_TRY"))) try;
156 /* The list through the catch handlers, the list of type objects
157 matched, and the list of associated filters. */
158 struct eh_region_u_catch
{
159 struct eh_region
*next_catch
;
160 struct eh_region
*prev_catch
;
163 } GTY ((tag ("ERT_CATCH"))) catch;
165 /* A tree_list of allowed types. */
166 struct eh_region_u_allowed
{
169 } GTY ((tag ("ERT_ALLOWED_EXCEPTIONS"))) allowed
;
171 /* The type given by a call to "throw foo();", or discovered
173 struct eh_region_u_throw
{
175 } GTY ((tag ("ERT_THROW"))) throw;
177 /* Retain the cleanup expression even after expansion so that
178 we can match up fixup regions. */
179 struct eh_region_u_cleanup
{
181 struct eh_region
*prev_try
;
182 } GTY ((tag ("ERT_CLEANUP"))) cleanup
;
184 /* The real region (by expression and by pointer) that fixup code
186 struct eh_region_u_fixup
{
188 struct eh_region
*real_region
;
190 } GTY ((tag ("ERT_FIXUP"))) fixup
;
191 } GTY ((desc ("%0.type"))) u
;
193 /* Entry point for this region's handler before landing pads are built. */
197 /* Entry point for this region's handler from the runtime eh library. */
200 /* Entry point for this region's handler from an inner region. */
201 rtx post_landing_pad
;
203 /* The RESX insn for handing off control to the next outermost handler,
207 /* True if something in this region may throw. */
208 unsigned may_contain_throw
: 1;
211 struct call_site_record
GTY(())
217 /* Used to save exception status for each function. */
218 struct eh_status
GTY(())
220 /* The tree of all regions for this function. */
221 struct eh_region
*region_tree
;
223 /* The same information as an indexable array. */
224 struct eh_region
** GTY ((length ("%h.last_region_number"))) region_array
;
226 /* The most recently open region. */
227 struct eh_region
*cur_region
;
229 /* This is the region for which we are processing catch blocks. */
230 struct eh_region
*try_region
;
235 int built_landing_pads
;
236 int last_region_number
;
238 varray_type ttype_data
;
239 varray_type ehspec_data
;
240 varray_type action_record_data
;
242 htab_t
GTY ((param_is (struct ehl_map_entry
))) exception_handler_label_map
;
244 struct call_site_record
* GTY ((length ("%h.call_site_data_used")))
246 int call_site_data_used
;
247 int call_site_data_size
;
258 static int t2r_eq (const void *, const void *);
259 static hashval_t
t2r_hash (const void *);
260 static void add_type_for_runtime (tree
);
261 static tree
lookup_type_for_runtime (tree
);
263 static void resolve_fixup_regions (void);
264 static void remove_fixup_regions (void);
265 static void remove_unreachable_regions (rtx
);
266 static void convert_from_eh_region_ranges_1 (rtx
*, int *, int);
268 static struct eh_region
*duplicate_eh_region_1 (struct eh_region
*,
269 struct inline_remap
*);
270 static void duplicate_eh_region_2 (struct eh_region
*, struct eh_region
**);
271 static int ttypes_filter_eq (const void *, const void *);
272 static hashval_t
ttypes_filter_hash (const void *);
273 static int ehspec_filter_eq (const void *, const void *);
274 static hashval_t
ehspec_filter_hash (const void *);
275 static int add_ttypes_entry (htab_t
, tree
);
276 static int add_ehspec_entry (htab_t
, htab_t
, tree
);
277 static void assign_filter_values (void);
278 static void build_post_landing_pads (void);
279 static void connect_post_landing_pads (void);
280 static void dw2_build_landing_pads (void);
283 static bool sjlj_find_directly_reachable_regions (struct sjlj_lp_info
*);
284 static void sjlj_assign_call_site_values (rtx
, struct sjlj_lp_info
*);
285 static void sjlj_mark_call_sites (struct sjlj_lp_info
*);
286 static void sjlj_emit_function_enter (rtx
);
287 static void sjlj_emit_function_exit (void);
288 static void sjlj_emit_dispatch_table (rtx
, struct sjlj_lp_info
*);
289 static void sjlj_build_landing_pads (void);
291 static hashval_t
ehl_hash (const void *);
292 static int ehl_eq (const void *, const void *);
293 static void add_ehl_entry (rtx
, struct eh_region
*);
294 static void remove_exception_handler_label (rtx
);
295 static void remove_eh_handler (struct eh_region
*);
296 static int for_each_eh_label_1 (void **, void *);
298 /* The return value of reachable_next_level. */
301 /* The given exception is not processed by the given region. */
303 /* The given exception may need processing by the given region. */
305 /* The given exception is completely processed by the given region. */
307 /* The given exception is completely processed by the runtime. */
311 struct reachable_info
;
312 static enum reachable_code
reachable_next_level (struct eh_region
*, tree
,
313 struct reachable_info
*);
315 static int action_record_eq (const void *, const void *);
316 static hashval_t
action_record_hash (const void *);
317 static int add_action_record (htab_t
, int, int);
318 static int collect_one_action_chain (htab_t
, struct eh_region
*);
319 static int add_call_site (rtx
, int);
321 static void push_uleb128 (varray_type
*, unsigned int);
322 static void push_sleb128 (varray_type
*, int);
323 #ifndef HAVE_AS_LEB128
324 static int dw2_size_of_call_site_table (void);
325 static int sjlj_size_of_call_site_table (void);
327 static void dw2_output_call_site_table (void);
328 static void sjlj_output_call_site_table (void);
331 /* Routine to see if exception handling is turned on.
332 DO_WARN is nonzero if we want to inform the user that exception
333 handling is turned off.
335 This is used to ensure that -fexceptions has been specified if the
336 compiler tries to use any exception-specific functions. */
339 doing_eh (int do_warn
)
341 if (! flag_exceptions
)
343 static int warned
= 0;
344 if (! warned
&& do_warn
)
346 error ("exception handling disabled, use -fexceptions to enable");
358 if (! flag_exceptions
)
361 type_to_runtime_map
= htab_create_ggc (31, t2r_hash
, t2r_eq
, NULL
);
363 /* Create the SjLj_Function_Context structure. This should match
364 the definition in unwind-sjlj.c. */
365 if (USING_SJLJ_EXCEPTIONS
)
367 tree f_jbuf
, f_per
, f_lsda
, f_prev
, f_cs
, f_data
, tmp
;
369 sjlj_fc_type_node
= lang_hooks
.types
.make_type (RECORD_TYPE
);
371 f_prev
= build_decl (FIELD_DECL
, get_identifier ("__prev"),
372 build_pointer_type (sjlj_fc_type_node
));
373 DECL_FIELD_CONTEXT (f_prev
) = sjlj_fc_type_node
;
375 f_cs
= build_decl (FIELD_DECL
, get_identifier ("__call_site"),
377 DECL_FIELD_CONTEXT (f_cs
) = sjlj_fc_type_node
;
379 tmp
= build_index_type (build_int_2 (4 - 1, 0));
380 tmp
= build_array_type (lang_hooks
.types
.type_for_mode (word_mode
, 1),
382 f_data
= build_decl (FIELD_DECL
, get_identifier ("__data"), tmp
);
383 DECL_FIELD_CONTEXT (f_data
) = sjlj_fc_type_node
;
385 f_per
= build_decl (FIELD_DECL
, get_identifier ("__personality"),
387 DECL_FIELD_CONTEXT (f_per
) = sjlj_fc_type_node
;
389 f_lsda
= build_decl (FIELD_DECL
, get_identifier ("__lsda"),
391 DECL_FIELD_CONTEXT (f_lsda
) = sjlj_fc_type_node
;
393 #ifdef DONT_USE_BUILTIN_SETJMP
395 tmp
= build_int_2 (JMP_BUF_SIZE
- 1, 0);
397 /* Should be large enough for most systems, if it is not,
398 JMP_BUF_SIZE should be defined with the proper value. It will
399 also tend to be larger than necessary for most systems, a more
400 optimal port will define JMP_BUF_SIZE. */
401 tmp
= build_int_2 (FIRST_PSEUDO_REGISTER
+ 2 - 1, 0);
404 /* builtin_setjmp takes a pointer to 5 words. */
405 tmp
= build_int_2 (5 * BITS_PER_WORD
/ POINTER_SIZE
- 1, 0);
407 tmp
= build_index_type (tmp
);
408 tmp
= build_array_type (ptr_type_node
, tmp
);
409 f_jbuf
= build_decl (FIELD_DECL
, get_identifier ("__jbuf"), tmp
);
410 #ifdef DONT_USE_BUILTIN_SETJMP
411 /* We don't know what the alignment requirements of the
412 runtime's jmp_buf has. Overestimate. */
413 DECL_ALIGN (f_jbuf
) = BIGGEST_ALIGNMENT
;
414 DECL_USER_ALIGN (f_jbuf
) = 1;
416 DECL_FIELD_CONTEXT (f_jbuf
) = sjlj_fc_type_node
;
418 TYPE_FIELDS (sjlj_fc_type_node
) = f_prev
;
419 TREE_CHAIN (f_prev
) = f_cs
;
420 TREE_CHAIN (f_cs
) = f_data
;
421 TREE_CHAIN (f_data
) = f_per
;
422 TREE_CHAIN (f_per
) = f_lsda
;
423 TREE_CHAIN (f_lsda
) = f_jbuf
;
425 layout_type (sjlj_fc_type_node
);
427 /* Cache the interesting field offsets so that we have
428 easy access from rtl. */
429 sjlj_fc_call_site_ofs
430 = (tree_low_cst (DECL_FIELD_OFFSET (f_cs
), 1)
431 + tree_low_cst (DECL_FIELD_BIT_OFFSET (f_cs
), 1) / BITS_PER_UNIT
);
433 = (tree_low_cst (DECL_FIELD_OFFSET (f_data
), 1)
434 + tree_low_cst (DECL_FIELD_BIT_OFFSET (f_data
), 1) / BITS_PER_UNIT
);
435 sjlj_fc_personality_ofs
436 = (tree_low_cst (DECL_FIELD_OFFSET (f_per
), 1)
437 + tree_low_cst (DECL_FIELD_BIT_OFFSET (f_per
), 1) / BITS_PER_UNIT
);
439 = (tree_low_cst (DECL_FIELD_OFFSET (f_lsda
), 1)
440 + tree_low_cst (DECL_FIELD_BIT_OFFSET (f_lsda
), 1) / BITS_PER_UNIT
);
442 = (tree_low_cst (DECL_FIELD_OFFSET (f_jbuf
), 1)
443 + tree_low_cst (DECL_FIELD_BIT_OFFSET (f_jbuf
), 1) / BITS_PER_UNIT
);
448 init_eh_for_function (void)
450 cfun
->eh
= ggc_alloc_cleared (sizeof (struct eh_status
));
453 /* Routines to generate the exception tree somewhat directly.
454 These are used from tree-eh.c when processing exception related
455 nodes during tree optimization. */
457 static struct eh_region
*
458 gen_eh_region (enum eh_region_type type
, struct eh_region
*outer
)
460 struct eh_region
*new;
462 #ifdef ENABLE_CHECKING
467 /* Insert a new blank region as a leaf in the tree. */
468 new = ggc_alloc_cleared (sizeof (*new));
473 new->next_peer
= outer
->inner
;
478 new->next_peer
= cfun
->eh
->region_tree
;
479 cfun
->eh
->region_tree
= new;
482 new->region_number
= ++cfun
->eh
->last_region_number
;
488 gen_eh_region_cleanup (struct eh_region
*outer
, struct eh_region
*prev_try
)
490 struct eh_region
*cleanup
= gen_eh_region (ERT_CLEANUP
, outer
);
491 cleanup
->u
.cleanup
.prev_try
= prev_try
;
496 gen_eh_region_try (struct eh_region
*outer
)
498 return gen_eh_region (ERT_TRY
, outer
);
502 gen_eh_region_catch (struct eh_region
*t
, tree type_or_list
)
504 struct eh_region
*c
, *l
;
505 tree type_list
, type_node
;
507 /* Ensure to always end up with a type list to normalize further
508 processing, then register each type against the runtime types map. */
509 type_list
= type_or_list
;
512 if (TREE_CODE (type_or_list
) != TREE_LIST
)
513 type_list
= tree_cons (NULL_TREE
, type_or_list
, NULL_TREE
);
515 type_node
= type_list
;
516 for (; type_node
; type_node
= TREE_CHAIN (type_node
))
517 add_type_for_runtime (TREE_VALUE (type_node
));
520 c
= gen_eh_region (ERT_CATCH
, t
->outer
);
521 c
->u
.catch.type_list
= type_list
;
522 l
= t
->u
.try.last_catch
;
523 c
->u
.catch.prev_catch
= l
;
525 l
->u
.catch.next_catch
= c
;
528 t
->u
.try.last_catch
= c
;
534 gen_eh_region_allowed (struct eh_region
*outer
, tree allowed
)
536 struct eh_region
*region
= gen_eh_region (ERT_ALLOWED_EXCEPTIONS
, outer
);
537 region
->u
.allowed
.type_list
= allowed
;
539 for (; allowed
; allowed
= TREE_CHAIN (allowed
))
540 add_type_for_runtime (TREE_VALUE (allowed
));
546 gen_eh_region_must_not_throw (struct eh_region
*outer
)
548 return gen_eh_region (ERT_MUST_NOT_THROW
, outer
);
552 get_eh_region_number (struct eh_region
*region
)
554 return region
->region_number
;
558 get_eh_region_may_contain_throw (struct eh_region
*region
)
560 return region
->may_contain_throw
;
564 get_eh_region_tree_label (struct eh_region
*region
)
566 return region
->tree_label
;
570 set_eh_region_tree_label (struct eh_region
*region
, tree lab
)
572 region
->tree_label
= lab
;
576 expand_resx_expr (tree exp
)
578 int region_nr
= TREE_INT_CST_LOW (TREE_OPERAND (exp
, 0));
579 struct eh_region
*reg
= cfun
->eh
->region_array
[region_nr
];
581 reg
->resume
= emit_jump_insn (gen_rtx_RESX (VOIDmode
, region_nr
));
585 /* Note that the current EH region (if any) may contain a throw, or a
586 call to a function which itself may contain a throw. */
589 note_eh_region_may_contain_throw (struct eh_region
*region
)
591 while (region
&& !region
->may_contain_throw
)
593 region
->may_contain_throw
= 1;
594 region
= region
->outer
;
599 note_current_region_may_contain_throw (void)
601 note_eh_region_may_contain_throw (cfun
->eh
->cur_region
);
605 /* Return an rtl expression for a pointer to the exception object
609 get_exception_pointer (struct function
*fun
)
611 rtx exc_ptr
= fun
->eh
->exc_ptr
;
612 if (fun
== cfun
&& ! exc_ptr
)
614 exc_ptr
= gen_reg_rtx (ptr_mode
);
615 fun
->eh
->exc_ptr
= exc_ptr
;
620 /* Return an rtl expression for the exception dispatch filter
624 get_exception_filter (struct function
*fun
)
626 rtx filter
= fun
->eh
->filter
;
627 if (fun
== cfun
&& ! filter
)
629 filter
= gen_reg_rtx (word_mode
);
630 fun
->eh
->filter
= filter
;
635 /* This section is for the exception handling specific optimization pass. */
637 /* Random access the exception region tree. */
640 collect_eh_region_array (void)
642 struct eh_region
**array
, *i
;
644 i
= cfun
->eh
->region_tree
;
648 array
= ggc_alloc_cleared ((cfun
->eh
->last_region_number
+ 1)
650 cfun
->eh
->region_array
= array
;
654 array
[i
->region_number
] = i
;
656 /* If there are sub-regions, process them. */
659 /* If there are peers, process them. */
660 else if (i
->next_peer
)
662 /* Otherwise, step back up the tree to the next peer. */
669 } while (i
->next_peer
== NULL
);
676 resolve_one_fixup_region (struct eh_region
*fixup
)
678 struct eh_region
*cleanup
, *real
;
681 n
= cfun
->eh
->last_region_number
;
684 for (j
= 1; j
<= n
; ++j
)
686 cleanup
= cfun
->eh
->region_array
[j
];
687 if (cleanup
&& cleanup
->type
== ERT_CLEANUP
688 && cleanup
->u
.cleanup
.exp
== fixup
->u
.fixup
.cleanup_exp
)
694 real
= cleanup
->outer
;
695 if (real
&& real
->type
== ERT_FIXUP
)
697 if (!real
->u
.fixup
.resolved
)
698 resolve_one_fixup_region (real
);
699 real
= real
->u
.fixup
.real_region
;
702 fixup
->u
.fixup
.real_region
= real
;
703 fixup
->u
.fixup
.resolved
= true;
707 resolve_fixup_regions (void)
709 int i
, n
= cfun
->eh
->last_region_number
;
711 for (i
= 1; i
<= n
; ++i
)
713 struct eh_region
*fixup
= cfun
->eh
->region_array
[i
];
715 if (!fixup
|| fixup
->type
!= ERT_FIXUP
|| fixup
->u
.fixup
.resolved
)
718 resolve_one_fixup_region (fixup
);
722 /* Now that we've discovered what region actually encloses a fixup,
723 we can shuffle pointers and remove them from the tree. */
726 remove_fixup_regions (void)
730 struct eh_region
*fixup
;
732 /* Walk the insn chain and adjust the REG_EH_REGION numbers
733 for instructions referencing fixup regions. This is only
734 strictly necessary for fixup regions with no parent, but
735 doesn't hurt to do it for all regions. */
736 for (insn
= get_insns(); insn
; insn
= NEXT_INSN (insn
))
738 && (note
= find_reg_note (insn
, REG_EH_REGION
, NULL
))
739 && INTVAL (XEXP (note
, 0)) > 0
740 && (fixup
= cfun
->eh
->region_array
[INTVAL (XEXP (note
, 0))])
741 && fixup
->type
== ERT_FIXUP
)
743 if (fixup
->u
.fixup
.real_region
)
744 XEXP (note
, 0) = GEN_INT (fixup
->u
.fixup
.real_region
->region_number
);
746 remove_note (insn
, note
);
749 /* Remove the fixup regions from the tree. */
750 for (i
= cfun
->eh
->last_region_number
; i
> 0; --i
)
752 fixup
= cfun
->eh
->region_array
[i
];
756 /* Allow GC to maybe free some memory. */
757 if (fixup
->type
== ERT_CLEANUP
)
758 fixup
->u
.cleanup
.exp
= NULL_TREE
;
760 if (fixup
->type
!= ERT_FIXUP
)
765 struct eh_region
*parent
, *p
, **pp
;
767 parent
= fixup
->u
.fixup
.real_region
;
769 /* Fix up the children's parent pointers; find the end of
771 for (p
= fixup
->inner
; ; p
= p
->next_peer
)
778 /* In the tree of cleanups, only outer-inner ordering matters.
779 So link the children back in anywhere at the correct level. */
783 pp
= &cfun
->eh
->region_tree
;
789 remove_eh_handler (fixup
);
793 /* Remove all regions whose labels are not reachable from insns. */
796 remove_unreachable_regions (rtx insns
)
798 int i
, *uid_region_num
;
803 uid_region_num
= xcalloc (get_max_uid (), sizeof(int));
804 reachable
= xcalloc (cfun
->eh
->last_region_number
+ 1, sizeof(bool));
806 for (i
= cfun
->eh
->last_region_number
; i
> 0; --i
)
808 r
= cfun
->eh
->region_array
[i
];
809 if (!r
|| r
->region_number
!= i
)
814 if (uid_region_num
[INSN_UID (r
->resume
)])
816 uid_region_num
[INSN_UID (r
->resume
)] = i
;
820 if (uid_region_num
[INSN_UID (r
->label
)])
822 uid_region_num
[INSN_UID (r
->label
)] = i
;
826 for (insn
= insns
; insn
; insn
= NEXT_INSN (insn
))
827 reachable
[uid_region_num
[INSN_UID (insn
)]] = true;
829 for (i
= cfun
->eh
->last_region_number
; i
> 0; --i
)
831 r
= cfun
->eh
->region_array
[i
];
832 if (r
&& r
->region_number
== i
&& !reachable
[i
])
838 /* Don't remove ERT_THROW regions if their outer region
840 if (r
->outer
&& reachable
[r
->outer
->region_number
])
844 case ERT_MUST_NOT_THROW
:
845 /* MUST_NOT_THROW regions are implementable solely in the
846 runtime, but their existence continues to affect calls
847 within that region. Never delete them here. */
853 /* TRY regions are reachable if any of its CATCH regions
856 for (c
= r
->u
.try.catch; c
; c
= c
->u
.catch.next_catch
)
857 if (reachable
[c
->region_number
])
870 remove_eh_handler (r
);
875 free (uid_region_num
);
878 /* Turn NOTE_INSN_EH_REGION notes into REG_EH_REGION notes for each
879 can_throw instruction in the region. */
882 convert_from_eh_region_ranges_1 (rtx
*pinsns
, int *orig_sp
, int cur
)
887 for (insn
= *pinsns
; insn
; insn
= next
)
889 next
= NEXT_INSN (insn
);
892 int kind
= NOTE_LINE_NUMBER (insn
);
893 if (kind
== NOTE_INSN_EH_REGION_BEG
894 || kind
== NOTE_INSN_EH_REGION_END
)
896 if (kind
== NOTE_INSN_EH_REGION_BEG
)
901 cur
= NOTE_EH_HANDLER (insn
);
903 r
= cfun
->eh
->region_array
[cur
];
904 if (r
->type
== ERT_FIXUP
)
906 r
= r
->u
.fixup
.real_region
;
907 cur
= r
? r
->region_number
: 0;
909 else if (r
->type
== ERT_CATCH
)
912 cur
= r
? r
->region_number
: 0;
924 else if (INSN_P (insn
))
927 && ! find_reg_note (insn
, REG_EH_REGION
, NULL_RTX
)
928 /* Calls can always potentially throw exceptions, unless
929 they have a REG_EH_REGION note with a value of 0 or less.
930 Which should be the only possible kind so far. */
932 /* If we wanted exceptions for non-call insns, then
933 any may_trap_p instruction could throw. */
934 || (flag_non_call_exceptions
935 && GET_CODE (PATTERN (insn
)) != CLOBBER
936 && GET_CODE (PATTERN (insn
)) != USE
937 && may_trap_p (PATTERN (insn
)))))
939 REG_NOTES (insn
) = alloc_EXPR_LIST (REG_EH_REGION
, GEN_INT (cur
),
950 collect_rtl_labels_from_trees (void)
952 int i
, n
= cfun
->eh
->last_region_number
;
953 for (i
= 1; i
<= n
; ++i
)
955 struct eh_region
*reg
= cfun
->eh
->region_array
[i
];
956 if (reg
&& reg
->tree_label
)
957 reg
->label
= DECL_RTL_IF_SET (reg
->tree_label
);
962 convert_from_eh_region_ranges (void)
964 rtx insns
= get_insns ();
966 if (cfun
->eh
->region_array
)
968 /* If the region array already exists, assume we're coming from
969 optimize_function_tree. In this case all we need to do is
970 collect the rtl labels that correspond to the tree labels
971 that we allocated earlier. */
972 collect_rtl_labels_from_trees ();
978 collect_eh_region_array ();
979 resolve_fixup_regions ();
981 stack
= xmalloc (sizeof (int) * (cfun
->eh
->last_region_number
+ 1));
982 convert_from_eh_region_ranges_1 (&insns
, stack
, 0);
985 remove_fixup_regions ();
988 remove_unreachable_regions (insns
);
992 add_ehl_entry (rtx label
, struct eh_region
*region
)
994 struct ehl_map_entry
**slot
, *entry
;
996 LABEL_PRESERVE_P (label
) = 1;
998 entry
= ggc_alloc (sizeof (*entry
));
999 entry
->label
= label
;
1000 entry
->region
= region
;
1002 slot
= (struct ehl_map_entry
**)
1003 htab_find_slot (cfun
->eh
->exception_handler_label_map
, entry
, INSERT
);
1005 /* Before landing pad creation, each exception handler has its own
1006 label. After landing pad creation, the exception handlers may
1007 share landing pads. This is ok, since maybe_remove_eh_handler
1008 only requires the 1-1 mapping before landing pad creation. */
1009 if (*slot
&& !cfun
->eh
->built_landing_pads
)
1016 find_exception_handler_labels (void)
1020 if (cfun
->eh
->exception_handler_label_map
)
1021 htab_empty (cfun
->eh
->exception_handler_label_map
);
1024 /* ??? The expansion factor here (3/2) must be greater than the htab
1025 occupancy factor (4/3) to avoid unnecessary resizing. */
1026 cfun
->eh
->exception_handler_label_map
1027 = htab_create_ggc (cfun
->eh
->last_region_number
* 3 / 2,
1028 ehl_hash
, ehl_eq
, NULL
);
1031 if (cfun
->eh
->region_tree
== NULL
)
1034 for (i
= cfun
->eh
->last_region_number
; i
> 0; --i
)
1036 struct eh_region
*region
= cfun
->eh
->region_array
[i
];
1039 if (! region
|| region
->region_number
!= i
)
1041 if (cfun
->eh
->built_landing_pads
)
1042 lab
= region
->landing_pad
;
1044 lab
= region
->label
;
1047 add_ehl_entry (lab
, region
);
1050 /* For sjlj exceptions, need the return label to remain live until
1051 after landing pad generation. */
1052 if (USING_SJLJ_EXCEPTIONS
&& ! cfun
->eh
->built_landing_pads
)
1053 add_ehl_entry (return_label
, NULL
);
1057 current_function_has_exception_handlers (void)
1061 for (i
= cfun
->eh
->last_region_number
; i
> 0; --i
)
1063 struct eh_region
*region
= cfun
->eh
->region_array
[i
];
1065 if (! region
|| region
->region_number
!= i
)
1067 if (region
->type
!= ERT_THROW
)
1074 static struct eh_region
*
1075 duplicate_eh_region_1 (struct eh_region
*o
, struct inline_remap
*map
)
1077 struct eh_region
*n
= ggc_alloc_cleared (sizeof (struct eh_region
));
1079 n
->region_number
= o
->region_number
+ cfun
->eh
->last_region_number
;
1085 case ERT_MUST_NOT_THROW
:
1089 if (o
->u
.try.continue_label
)
1090 n
->u
.try.continue_label
1091 = get_label_from_map (map
,
1092 CODE_LABEL_NUMBER (o
->u
.try.continue_label
));
1096 n
->u
.catch.type_list
= o
->u
.catch.type_list
;
1099 case ERT_ALLOWED_EXCEPTIONS
:
1100 n
->u
.allowed
.type_list
= o
->u
.allowed
.type_list
;
1104 n
->u
.throw.type
= o
->u
.throw.type
;
1111 n
->label
= get_label_from_map (map
, CODE_LABEL_NUMBER (o
->label
));
1114 n
->resume
= map
->insn_map
[INSN_UID (o
->resume
)];
1115 if (n
->resume
== NULL
)
1123 duplicate_eh_region_2 (struct eh_region
*o
, struct eh_region
**n_array
)
1125 struct eh_region
*n
= n_array
[o
->region_number
];
1130 n
->u
.try.catch = n_array
[o
->u
.try.catch->region_number
];
1131 n
->u
.try.last_catch
= n_array
[o
->u
.try.last_catch
->region_number
];
1135 if (o
->u
.catch.next_catch
)
1136 n
->u
.catch.next_catch
= n_array
[o
->u
.catch.next_catch
->region_number
];
1137 if (o
->u
.catch.prev_catch
)
1138 n
->u
.catch.prev_catch
= n_array
[o
->u
.catch.prev_catch
->region_number
];
1146 n
->outer
= n_array
[o
->outer
->region_number
];
1148 n
->inner
= n_array
[o
->inner
->region_number
];
1150 n
->next_peer
= n_array
[o
->next_peer
->region_number
];
1154 duplicate_eh_regions (struct function
*ifun
, struct inline_remap
*map
)
1156 int ifun_last_region_number
= ifun
->eh
->last_region_number
;
1157 struct eh_region
**n_array
, *root
, *cur
;
1160 if (ifun_last_region_number
== 0)
1163 n_array
= xcalloc (ifun_last_region_number
+ 1, sizeof (*n_array
));
1165 for (i
= 1; i
<= ifun_last_region_number
; ++i
)
1167 cur
= ifun
->eh
->region_array
[i
];
1168 if (!cur
|| cur
->region_number
!= i
)
1170 n_array
[i
] = duplicate_eh_region_1 (cur
, map
);
1172 for (i
= 1; i
<= ifun_last_region_number
; ++i
)
1174 cur
= ifun
->eh
->region_array
[i
];
1175 if (!cur
|| cur
->region_number
!= i
)
1177 duplicate_eh_region_2 (cur
, n_array
);
1180 root
= n_array
[ifun
->eh
->region_tree
->region_number
];
1181 cur
= cfun
->eh
->cur_region
;
1184 struct eh_region
*p
= cur
->inner
;
1187 while (p
->next_peer
)
1189 p
->next_peer
= root
;
1194 for (i
= 1; i
<= ifun_last_region_number
; ++i
)
1195 if (n_array
[i
] && n_array
[i
]->outer
== NULL
)
1196 n_array
[i
]->outer
= cur
;
1200 struct eh_region
*p
= cfun
->eh
->region_tree
;
1203 while (p
->next_peer
)
1205 p
->next_peer
= root
;
1208 cfun
->eh
->region_tree
= root
;
1213 i
= cfun
->eh
->last_region_number
;
1214 cfun
->eh
->last_region_number
= i
+ ifun_last_region_number
;
1220 t2r_eq (const void *pentry
, const void *pdata
)
1222 tree entry
= (tree
) pentry
;
1223 tree data
= (tree
) pdata
;
1225 return TREE_PURPOSE (entry
) == data
;
1229 t2r_hash (const void *pentry
)
1231 tree entry
= (tree
) pentry
;
1232 return TREE_HASH (TREE_PURPOSE (entry
));
1236 add_type_for_runtime (tree type
)
1240 slot
= (tree
*) htab_find_slot_with_hash (type_to_runtime_map
, type
,
1241 TREE_HASH (type
), INSERT
);
1244 tree runtime
= (*lang_eh_runtime_type
) (type
);
1245 *slot
= tree_cons (type
, runtime
, NULL_TREE
);
1250 lookup_type_for_runtime (tree type
)
1254 slot
= (tree
*) htab_find_slot_with_hash (type_to_runtime_map
, type
,
1255 TREE_HASH (type
), NO_INSERT
);
1257 /* We should have always inserted the data earlier. */
1258 return TREE_VALUE (*slot
);
1262 /* Represent an entry in @TTypes for either catch actions
1263 or exception filter actions. */
1264 struct ttypes_filter
GTY(())
1270 /* Compare ENTRY (a ttypes_filter entry in the hash table) with DATA
1271 (a tree) for a @TTypes type node we are thinking about adding. */
1274 ttypes_filter_eq (const void *pentry
, const void *pdata
)
1276 const struct ttypes_filter
*entry
= (const struct ttypes_filter
*) pentry
;
1277 tree data
= (tree
) pdata
;
1279 return entry
->t
== data
;
1283 ttypes_filter_hash (const void *pentry
)
1285 const struct ttypes_filter
*entry
= (const struct ttypes_filter
*) pentry
;
1286 return TREE_HASH (entry
->t
);
1289 /* Compare ENTRY with DATA (both struct ttypes_filter) for a @TTypes
1290 exception specification list we are thinking about adding. */
1291 /* ??? Currently we use the type lists in the order given. Someone
1292 should put these in some canonical order. */
1295 ehspec_filter_eq (const void *pentry
, const void *pdata
)
1297 const struct ttypes_filter
*entry
= (const struct ttypes_filter
*) pentry
;
1298 const struct ttypes_filter
*data
= (const struct ttypes_filter
*) pdata
;
1300 return type_list_equal (entry
->t
, data
->t
);
1303 /* Hash function for exception specification lists. */
1306 ehspec_filter_hash (const void *pentry
)
1308 const struct ttypes_filter
*entry
= (const struct ttypes_filter
*) pentry
;
1312 for (list
= entry
->t
; list
; list
= TREE_CHAIN (list
))
1313 h
= (h
<< 5) + (h
>> 27) + TREE_HASH (TREE_VALUE (list
));
1317 /* Add TYPE (which may be NULL) to cfun->eh->ttype_data, using TYPES_HASH
1318 to speed up the search. Return the filter value to be used. */
1321 add_ttypes_entry (htab_t ttypes_hash
, tree type
)
1323 struct ttypes_filter
**slot
, *n
;
1325 slot
= (struct ttypes_filter
**)
1326 htab_find_slot_with_hash (ttypes_hash
, type
, TREE_HASH (type
), INSERT
);
1328 if ((n
= *slot
) == NULL
)
1330 /* Filter value is a 1 based table index. */
1332 n
= xmalloc (sizeof (*n
));
1334 n
->filter
= VARRAY_ACTIVE_SIZE (cfun
->eh
->ttype_data
) + 1;
1337 VARRAY_PUSH_TREE (cfun
->eh
->ttype_data
, type
);
1343 /* Add LIST to cfun->eh->ehspec_data, using EHSPEC_HASH and TYPES_HASH
1344 to speed up the search. Return the filter value to be used. */
1347 add_ehspec_entry (htab_t ehspec_hash
, htab_t ttypes_hash
, tree list
)
1349 struct ttypes_filter
**slot
, *n
;
1350 struct ttypes_filter dummy
;
1353 slot
= (struct ttypes_filter
**)
1354 htab_find_slot (ehspec_hash
, &dummy
, INSERT
);
1356 if ((n
= *slot
) == NULL
)
1358 /* Filter value is a -1 based byte index into a uleb128 buffer. */
1360 n
= xmalloc (sizeof (*n
));
1362 n
->filter
= -(VARRAY_ACTIVE_SIZE (cfun
->eh
->ehspec_data
) + 1);
1365 /* Look up each type in the list and encode its filter
1366 value as a uleb128. Terminate the list with 0. */
1367 for (; list
; list
= TREE_CHAIN (list
))
1368 push_uleb128 (&cfun
->eh
->ehspec_data
,
1369 add_ttypes_entry (ttypes_hash
, TREE_VALUE (list
)));
1370 VARRAY_PUSH_UCHAR (cfun
->eh
->ehspec_data
, 0);
1376 /* Generate the action filter values to be used for CATCH and
1377 ALLOWED_EXCEPTIONS regions. When using dwarf2 exception regions,
1378 we use lots of landing pads, and so every type or list can share
1379 the same filter value, which saves table space. */
1382 assign_filter_values (void)
1385 htab_t ttypes
, ehspec
;
1387 VARRAY_TREE_INIT (cfun
->eh
->ttype_data
, 16, "ttype_data");
1388 VARRAY_UCHAR_INIT (cfun
->eh
->ehspec_data
, 64, "ehspec_data");
1390 ttypes
= htab_create (31, ttypes_filter_hash
, ttypes_filter_eq
, free
);
1391 ehspec
= htab_create (31, ehspec_filter_hash
, ehspec_filter_eq
, free
);
1393 for (i
= cfun
->eh
->last_region_number
; i
> 0; --i
)
1395 struct eh_region
*r
= cfun
->eh
->region_array
[i
];
1397 /* Mind we don't process a region more than once. */
1398 if (!r
|| r
->region_number
!= i
)
1404 /* Whatever type_list is (NULL or true list), we build a list
1405 of filters for the region. */
1406 r
->u
.catch.filter_list
= NULL_TREE
;
1408 if (r
->u
.catch.type_list
!= NULL
)
1410 /* Get a filter value for each of the types caught and store
1411 them in the region's dedicated list. */
1412 tree tp_node
= r
->u
.catch.type_list
;
1414 for (;tp_node
; tp_node
= TREE_CHAIN (tp_node
))
1416 int flt
= add_ttypes_entry (ttypes
, TREE_VALUE (tp_node
));
1417 tree flt_node
= build_int_2 (flt
, 0);
1419 r
->u
.catch.filter_list
1420 = tree_cons (NULL_TREE
, flt_node
, r
->u
.catch.filter_list
);
1425 /* Get a filter value for the NULL list also since it will need
1426 an action record anyway. */
1427 int flt
= add_ttypes_entry (ttypes
, NULL
);
1428 tree flt_node
= build_int_2 (flt
, 0);
1430 r
->u
.catch.filter_list
1431 = tree_cons (NULL_TREE
, flt_node
, r
->u
.catch.filter_list
);
1436 case ERT_ALLOWED_EXCEPTIONS
:
1438 = add_ehspec_entry (ehspec
, ttypes
, r
->u
.allowed
.type_list
);
1446 htab_delete (ttypes
);
1447 htab_delete (ehspec
);
1450 /* Emit SEQ into basic block just before INSN (that is assumed to be
1451 first instruction of some existing BB and return the newly
1454 emit_to_new_bb_before (rtx seq
, rtx insn
)
1460 /* If there happens to be an fallthru edge (possibly created by cleanup_cfg
1461 call), we don't want it to go into newly created landing pad or other EH
1463 for (e
= BLOCK_FOR_INSN (insn
)->pred
; e
; e
= e
->pred_next
)
1464 if (e
->flags
& EDGE_FALLTHRU
)
1465 force_nonfallthru (e
);
1466 last
= emit_insn_before (seq
, insn
);
1467 if (BARRIER_P (last
))
1468 last
= PREV_INSN (last
);
1469 bb
= create_basic_block (seq
, last
, BLOCK_FOR_INSN (insn
)->prev_bb
);
1470 update_bb_for_insn (bb
);
1471 bb
->flags
|= BB_SUPERBLOCK
;
1475 /* Generate the code to actually handle exceptions, which will follow the
1479 build_post_landing_pads (void)
1483 for (i
= cfun
->eh
->last_region_number
; i
> 0; --i
)
1485 struct eh_region
*region
= cfun
->eh
->region_array
[i
];
1488 /* Mind we don't process a region more than once. */
1489 if (!region
|| region
->region_number
!= i
)
1492 switch (region
->type
)
1495 /* ??? Collect the set of all non-overlapping catch handlers
1496 all the way up the chain until blocked by a cleanup. */
1497 /* ??? Outer try regions can share landing pads with inner
1498 try regions if the types are completely non-overlapping,
1499 and there are no intervening cleanups. */
1501 region
->post_landing_pad
= gen_label_rtx ();
1505 emit_label (region
->post_landing_pad
);
1507 /* ??? It is mighty inconvenient to call back into the
1508 switch statement generation code in expand_end_case.
1509 Rapid prototyping sez a sequence of ifs. */
1511 struct eh_region
*c
;
1512 for (c
= region
->u
.try.catch; c
; c
= c
->u
.catch.next_catch
)
1514 if (c
->u
.catch.type_list
== NULL
)
1515 emit_jump (c
->label
);
1518 /* Need for one cmp/jump per type caught. Each type
1519 list entry has a matching entry in the filter list
1520 (see assign_filter_values). */
1521 tree tp_node
= c
->u
.catch.type_list
;
1522 tree flt_node
= c
->u
.catch.filter_list
;
1526 emit_cmp_and_jump_insns
1528 GEN_INT (tree_low_cst (TREE_VALUE (flt_node
), 0)),
1529 EQ
, NULL_RTX
, word_mode
, 0, c
->label
);
1531 tp_node
= TREE_CHAIN (tp_node
);
1532 flt_node
= TREE_CHAIN (flt_node
);
1538 /* We delay the generation of the _Unwind_Resume until we generate
1539 landing pads. We emit a marker here so as to get good control
1540 flow data in the meantime. */
1542 = emit_jump_insn (gen_rtx_RESX (VOIDmode
, region
->region_number
));
1548 emit_to_new_bb_before (seq
, region
->u
.try.catch->label
);
1552 case ERT_ALLOWED_EXCEPTIONS
:
1553 region
->post_landing_pad
= gen_label_rtx ();
1557 emit_label (region
->post_landing_pad
);
1559 emit_cmp_and_jump_insns (cfun
->eh
->filter
,
1560 GEN_INT (region
->u
.allowed
.filter
),
1561 EQ
, NULL_RTX
, word_mode
, 0, region
->label
);
1563 /* We delay the generation of the _Unwind_Resume until we generate
1564 landing pads. We emit a marker here so as to get good control
1565 flow data in the meantime. */
1567 = emit_jump_insn (gen_rtx_RESX (VOIDmode
, region
->region_number
));
1573 emit_to_new_bb_before (seq
, region
->label
);
1577 case ERT_MUST_NOT_THROW
:
1578 region
->post_landing_pad
= region
->label
;
1583 /* Nothing to do. */
1592 /* Replace RESX patterns with jumps to the next handler if any, or calls to
1593 _Unwind_Resume otherwise. */
1596 connect_post_landing_pads (void)
1600 for (i
= cfun
->eh
->last_region_number
; i
> 0; --i
)
1602 struct eh_region
*region
= cfun
->eh
->region_array
[i
];
1603 struct eh_region
*outer
;
1607 /* Mind we don't process a region more than once. */
1608 if (!region
|| region
->region_number
!= i
)
1611 /* If there is no RESX, or it has been deleted by flow, there's
1612 nothing to fix up. */
1613 if (! region
->resume
|| INSN_DELETED_P (region
->resume
))
1616 /* Search for another landing pad in this function. */
1617 for (outer
= region
->outer
; outer
; outer
= outer
->outer
)
1618 if (outer
->post_landing_pad
)
1626 basic_block src
, dest
;
1628 emit_jump (outer
->post_landing_pad
);
1629 src
= BLOCK_FOR_INSN (region
->resume
);
1630 dest
= BLOCK_FOR_INSN (outer
->post_landing_pad
);
1632 remove_edge (src
->succ
);
1633 e
= make_edge (src
, dest
, 0);
1634 e
->probability
= REG_BR_PROB_BASE
;
1635 e
->count
= src
->count
;
1639 emit_library_call (unwind_resume_libfunc
, LCT_THROW
,
1640 VOIDmode
, 1, cfun
->eh
->exc_ptr
, ptr_mode
);
1642 /* What we just emitted was a throwing libcall, so it got a
1643 barrier automatically added after it. If the last insn in
1644 the libcall sequence isn't the barrier, it's because the
1645 target emits multiple insns for a call, and there are insns
1646 after the actual call insn (which are redundant and would be
1647 optimized away). The barrier is inserted exactly after the
1648 call insn, so let's go get that and delete the insns after
1649 it, because below we need the barrier to be the last insn in
1651 delete_insns_since (NEXT_INSN (last_call_insn ()));
1656 barrier
= emit_insn_before (seq
, region
->resume
);
1657 /* Avoid duplicate barrier. */
1658 if (!BARRIER_P (barrier
))
1660 delete_insn (barrier
);
1661 delete_insn (region
->resume
);
1663 /* ??? From tree-ssa we can wind up with catch regions whose
1664 label is not instantiated, but whose resx is present. Now
1665 that we've dealt with the resx, kill the region. */
1666 if (region
->label
== NULL
&& region
->type
== ERT_CLEANUP
)
1667 remove_eh_handler (region
);
1673 dw2_build_landing_pads (void)
1678 for (i
= cfun
->eh
->last_region_number
; i
> 0; --i
)
1680 struct eh_region
*region
= cfun
->eh
->region_array
[i
];
1683 bool clobbers_hard_regs
= false;
1686 /* Mind we don't process a region more than once. */
1687 if (!region
|| region
->region_number
!= i
)
1690 if (region
->type
!= ERT_CLEANUP
1691 && region
->type
!= ERT_TRY
1692 && region
->type
!= ERT_ALLOWED_EXCEPTIONS
)
1697 region
->landing_pad
= gen_label_rtx ();
1698 emit_label (region
->landing_pad
);
1700 #ifdef HAVE_exception_receiver
1701 if (HAVE_exception_receiver
)
1702 emit_insn (gen_exception_receiver ());
1705 #ifdef HAVE_nonlocal_goto_receiver
1706 if (HAVE_nonlocal_goto_receiver
)
1707 emit_insn (gen_nonlocal_goto_receiver ());
1712 /* If the eh_return data registers are call-saved, then we
1713 won't have considered them clobbered from the call that
1714 threw. Kill them now. */
1717 unsigned r
= EH_RETURN_DATA_REGNO (j
);
1718 if (r
== INVALID_REGNUM
)
1720 if (! call_used_regs
[r
])
1722 emit_insn (gen_rtx_CLOBBER (VOIDmode
, gen_rtx_REG (Pmode
, r
)));
1723 clobbers_hard_regs
= true;
1727 if (clobbers_hard_regs
)
1729 /* @@@ This is a kludge. Not all machine descriptions define a
1730 blockage insn, but we must not allow the code we just generated
1731 to be reordered by scheduling. So emit an ASM_INPUT to act as
1733 emit_insn (gen_rtx_ASM_INPUT (VOIDmode
, ""));
1736 emit_move_insn (cfun
->eh
->exc_ptr
,
1737 gen_rtx_REG (ptr_mode
, EH_RETURN_DATA_REGNO (0)));
1738 emit_move_insn (cfun
->eh
->filter
,
1739 gen_rtx_REG (word_mode
, EH_RETURN_DATA_REGNO (1)));
1744 bb
= emit_to_new_bb_before (seq
, region
->post_landing_pad
);
1745 e
= make_edge (bb
, bb
->next_bb
, EDGE_FALLTHRU
);
1746 e
->count
= bb
->count
;
1747 e
->probability
= REG_BR_PROB_BASE
;
1754 int directly_reachable
;
1757 int call_site_index
;
1761 sjlj_find_directly_reachable_regions (struct sjlj_lp_info
*lp_info
)
1764 bool found_one
= false;
1766 for (insn
= get_insns (); insn
; insn
= NEXT_INSN (insn
))
1768 struct eh_region
*region
;
1769 enum reachable_code rc
;
1773 if (! INSN_P (insn
))
1776 note
= find_reg_note (insn
, REG_EH_REGION
, NULL_RTX
);
1777 if (!note
|| INTVAL (XEXP (note
, 0)) <= 0)
1780 region
= cfun
->eh
->region_array
[INTVAL (XEXP (note
, 0))];
1782 type_thrown
= NULL_TREE
;
1783 if (region
->type
== ERT_THROW
)
1785 type_thrown
= region
->u
.throw.type
;
1786 region
= region
->outer
;
1789 /* Find the first containing region that might handle the exception.
1790 That's the landing pad to which we will transfer control. */
1791 rc
= RNL_NOT_CAUGHT
;
1792 for (; region
; region
= region
->outer
)
1794 rc
= reachable_next_level (region
, type_thrown
, NULL
);
1795 if (rc
!= RNL_NOT_CAUGHT
)
1798 if (rc
== RNL_MAYBE_CAUGHT
|| rc
== RNL_CAUGHT
)
1800 lp_info
[region
->region_number
].directly_reachable
= 1;
1809 sjlj_assign_call_site_values (rtx dispatch_label
, struct sjlj_lp_info
*lp_info
)
1814 /* First task: build the action table. */
1816 VARRAY_UCHAR_INIT (cfun
->eh
->action_record_data
, 64, "action_record_data");
1817 ar_hash
= htab_create (31, action_record_hash
, action_record_eq
, free
);
1819 for (i
= cfun
->eh
->last_region_number
; i
> 0; --i
)
1820 if (lp_info
[i
].directly_reachable
)
1822 struct eh_region
*r
= cfun
->eh
->region_array
[i
];
1823 r
->landing_pad
= dispatch_label
;
1824 lp_info
[i
].action_index
= collect_one_action_chain (ar_hash
, r
);
1825 if (lp_info
[i
].action_index
!= -1)
1826 cfun
->uses_eh_lsda
= 1;
1829 htab_delete (ar_hash
);
1831 /* Next: assign dispatch values. In dwarf2 terms, this would be the
1832 landing pad label for the region. For sjlj though, there is one
1833 common landing pad from which we dispatch to the post-landing pads.
1835 A region receives a dispatch index if it is directly reachable
1836 and requires in-function processing. Regions that share post-landing
1837 pads may share dispatch indices. */
1838 /* ??? Post-landing pad sharing doesn't actually happen at the moment
1839 (see build_post_landing_pads) so we don't bother checking for it. */
1842 for (i
= cfun
->eh
->last_region_number
; i
> 0; --i
)
1843 if (lp_info
[i
].directly_reachable
)
1844 lp_info
[i
].dispatch_index
= index
++;
1846 /* Finally: assign call-site values. If dwarf2 terms, this would be
1847 the region number assigned by convert_to_eh_region_ranges, but
1848 handles no-action and must-not-throw differently. */
1851 for (i
= cfun
->eh
->last_region_number
; i
> 0; --i
)
1852 if (lp_info
[i
].directly_reachable
)
1854 int action
= lp_info
[i
].action_index
;
1856 /* Map must-not-throw to otherwise unused call-site index 0. */
1859 /* Map no-action to otherwise unused call-site index -1. */
1860 else if (action
== -1)
1862 /* Otherwise, look it up in the table. */
1864 index
= add_call_site (GEN_INT (lp_info
[i
].dispatch_index
), action
);
1866 lp_info
[i
].call_site_index
= index
;
1871 sjlj_mark_call_sites (struct sjlj_lp_info
*lp_info
)
1873 int last_call_site
= -2;
1876 for (insn
= get_insns (); insn
; insn
= NEXT_INSN (insn
))
1878 struct eh_region
*region
;
1880 rtx note
, before
, p
;
1882 /* Reset value tracking at extended basic block boundaries. */
1884 last_call_site
= -2;
1886 if (! INSN_P (insn
))
1889 note
= find_reg_note (insn
, REG_EH_REGION
, NULL_RTX
);
1892 /* Calls (and trapping insns) without notes are outside any
1893 exception handling region in this function. Mark them as
1896 || (flag_non_call_exceptions
1897 && may_trap_p (PATTERN (insn
))))
1898 this_call_site
= -1;
1904 /* Calls that are known to not throw need not be marked. */
1905 if (INTVAL (XEXP (note
, 0)) <= 0)
1908 region
= cfun
->eh
->region_array
[INTVAL (XEXP (note
, 0))];
1909 this_call_site
= lp_info
[region
->region_number
].call_site_index
;
1912 if (this_call_site
== last_call_site
)
1915 /* Don't separate a call from it's argument loads. */
1918 before
= find_first_parameter_load (insn
, NULL_RTX
);
1921 mem
= adjust_address (cfun
->eh
->sjlj_fc
, TYPE_MODE (integer_type_node
),
1922 sjlj_fc_call_site_ofs
);
1923 emit_move_insn (mem
, GEN_INT (this_call_site
));
1927 emit_insn_before (p
, before
);
1928 last_call_site
= this_call_site
;
1932 /* Construct the SjLj_Function_Context. */
1935 sjlj_emit_function_enter (rtx dispatch_label
)
1937 rtx fn_begin
, fc
, mem
, seq
;
1939 fc
= cfun
->eh
->sjlj_fc
;
1943 /* We're storing this libcall's address into memory instead of
1944 calling it directly. Thus, we must call assemble_external_libcall
1945 here, as we can not depend on emit_library_call to do it for us. */
1946 assemble_external_libcall (eh_personality_libfunc
);
1947 mem
= adjust_address (fc
, Pmode
, sjlj_fc_personality_ofs
);
1948 emit_move_insn (mem
, eh_personality_libfunc
);
1950 mem
= adjust_address (fc
, Pmode
, sjlj_fc_lsda_ofs
);
1951 if (cfun
->uses_eh_lsda
)
1956 ASM_GENERATE_INTERNAL_LABEL (buf
, "LLSDA", current_function_funcdef_no
);
1957 sym
= gen_rtx_SYMBOL_REF (Pmode
, ggc_strdup (buf
));
1958 SYMBOL_REF_FLAGS (sym
) = SYMBOL_FLAG_LOCAL
;
1959 emit_move_insn (mem
, sym
);
1962 emit_move_insn (mem
, const0_rtx
);
1964 #ifdef DONT_USE_BUILTIN_SETJMP
1967 x
= emit_library_call_value (setjmp_libfunc
, NULL_RTX
, LCT_RETURNS_TWICE
,
1968 TYPE_MODE (integer_type_node
), 1,
1969 plus_constant (XEXP (fc
, 0),
1970 sjlj_fc_jbuf_ofs
), Pmode
);
1972 note
= emit_note (NOTE_INSN_EXPECTED_VALUE
);
1973 NOTE_EXPECTED_VALUE (note
) = gen_rtx_EQ (VOIDmode
, x
, const0_rtx
);
1975 emit_cmp_and_jump_insns (x
, const0_rtx
, NE
, 0,
1976 TYPE_MODE (integer_type_node
), 0, dispatch_label
);
1979 expand_builtin_setjmp_setup (plus_constant (XEXP (fc
, 0), sjlj_fc_jbuf_ofs
),
1983 emit_library_call (unwind_sjlj_register_libfunc
, LCT_NORMAL
, VOIDmode
,
1984 1, XEXP (fc
, 0), Pmode
);
1989 /* ??? Instead of doing this at the beginning of the function,
1990 do this in a block that is at loop level 0 and dominates all
1991 can_throw_internal instructions. */
1993 for (fn_begin
= get_insns (); ; fn_begin
= NEXT_INSN (fn_begin
))
1994 if (NOTE_P (fn_begin
)
1995 && (NOTE_LINE_NUMBER (fn_begin
) == NOTE_INSN_FUNCTION_BEG
1996 || NOTE_LINE_NUMBER (fn_begin
) == NOTE_INSN_BASIC_BLOCK
))
1998 if (NOTE_LINE_NUMBER (fn_begin
) == NOTE_INSN_FUNCTION_BEG
)
1999 insert_insn_on_edge (seq
, ENTRY_BLOCK_PTR
->succ
);
2002 rtx last
= BB_END (ENTRY_BLOCK_PTR
->succ
->dest
);
2003 for (; ; fn_begin
= NEXT_INSN (fn_begin
))
2004 if ((NOTE_P (fn_begin
)
2005 && NOTE_LINE_NUMBER (fn_begin
) == NOTE_INSN_FUNCTION_BEG
)
2006 || fn_begin
== last
)
2008 emit_insn_after (seq
, fn_begin
);
2012 /* Call back from expand_function_end to know where we should put
2013 the call to unwind_sjlj_unregister_libfunc if needed. */
2016 sjlj_emit_function_exit_after (rtx after
)
2018 cfun
->eh
->sjlj_exit_after
= after
;
2022 sjlj_emit_function_exit (void)
2029 emit_library_call (unwind_sjlj_unregister_libfunc
, LCT_NORMAL
, VOIDmode
,
2030 1, XEXP (cfun
->eh
->sjlj_fc
, 0), Pmode
);
2035 /* ??? Really this can be done in any block at loop level 0 that
2036 post-dominates all can_throw_internal instructions. This is
2037 the last possible moment. */
2039 for (e
= EXIT_BLOCK_PTR
->pred
; e
; e
= e
->pred_next
)
2040 if (e
->flags
& EDGE_FALLTHRU
)
2046 /* Figure out whether the place we are supposed to insert libcall
2047 is inside the last basic block or after it. In the other case
2048 we need to emit to edge. */
2049 if (e
->src
->next_bb
!= EXIT_BLOCK_PTR
)
2051 for (insn
= NEXT_INSN (BB_END (e
->src
)); insn
; insn
= NEXT_INSN (insn
))
2052 if (insn
== cfun
->eh
->sjlj_exit_after
)
2055 insert_insn_on_edge (seq
, e
);
2058 insn
= cfun
->eh
->sjlj_exit_after
;
2060 insn
= NEXT_INSN (insn
);
2061 emit_insn_after (seq
, insn
);
2067 sjlj_emit_dispatch_table (rtx dispatch_label
, struct sjlj_lp_info
*lp_info
)
2069 int i
, first_reachable
;
2070 rtx mem
, dispatch
, seq
, fc
;
2075 fc
= cfun
->eh
->sjlj_fc
;
2079 emit_label (dispatch_label
);
2081 #ifndef DONT_USE_BUILTIN_SETJMP
2082 expand_builtin_setjmp_receiver (dispatch_label
);
2085 /* Load up dispatch index, exc_ptr and filter values from the
2086 function context. */
2087 mem
= adjust_address (fc
, TYPE_MODE (integer_type_node
),
2088 sjlj_fc_call_site_ofs
);
2089 dispatch
= copy_to_reg (mem
);
2091 mem
= adjust_address (fc
, word_mode
, sjlj_fc_data_ofs
);
2092 if (word_mode
!= ptr_mode
)
2094 #ifdef POINTERS_EXTEND_UNSIGNED
2095 mem
= convert_memory_address (ptr_mode
, mem
);
2097 mem
= convert_to_mode (ptr_mode
, mem
, 0);
2100 emit_move_insn (cfun
->eh
->exc_ptr
, mem
);
2102 mem
= adjust_address (fc
, word_mode
, sjlj_fc_data_ofs
+ UNITS_PER_WORD
);
2103 emit_move_insn (cfun
->eh
->filter
, mem
);
2105 /* Jump to one of the directly reachable regions. */
2106 /* ??? This really ought to be using a switch statement. */
2108 first_reachable
= 0;
2109 for (i
= cfun
->eh
->last_region_number
; i
> 0; --i
)
2111 if (! lp_info
[i
].directly_reachable
)
2114 if (! first_reachable
)
2116 first_reachable
= i
;
2120 emit_cmp_and_jump_insns (dispatch
, GEN_INT (lp_info
[i
].dispatch_index
),
2121 EQ
, NULL_RTX
, TYPE_MODE (integer_type_node
), 0,
2122 cfun
->eh
->region_array
[i
]->post_landing_pad
);
2128 before
= cfun
->eh
->region_array
[first_reachable
]->post_landing_pad
;
2130 bb
= emit_to_new_bb_before (seq
, before
);
2131 e
= make_edge (bb
, bb
->next_bb
, EDGE_FALLTHRU
);
2132 e
->count
= bb
->count
;
2133 e
->probability
= REG_BR_PROB_BASE
;
2137 sjlj_build_landing_pads (void)
2139 struct sjlj_lp_info
*lp_info
;
2141 lp_info
= xcalloc (cfun
->eh
->last_region_number
+ 1,
2142 sizeof (struct sjlj_lp_info
));
2144 if (sjlj_find_directly_reachable_regions (lp_info
))
2146 rtx dispatch_label
= gen_label_rtx ();
2149 = assign_stack_local (TYPE_MODE (sjlj_fc_type_node
),
2150 int_size_in_bytes (sjlj_fc_type_node
),
2151 TYPE_ALIGN (sjlj_fc_type_node
));
2153 sjlj_assign_call_site_values (dispatch_label
, lp_info
);
2154 sjlj_mark_call_sites (lp_info
);
2156 sjlj_emit_function_enter (dispatch_label
);
2157 sjlj_emit_dispatch_table (dispatch_label
, lp_info
);
2158 sjlj_emit_function_exit ();
2165 finish_eh_generation (void)
2169 /* Nothing to do if no regions created. */
2170 if (cfun
->eh
->region_tree
== NULL
)
2173 /* The object here is to provide find_basic_blocks with detailed
2174 information (via reachable_handlers) on how exception control
2175 flows within the function. In this first pass, we can include
2176 type information garnered from ERT_THROW and ERT_ALLOWED_EXCEPTIONS
2177 regions, and hope that it will be useful in deleting unreachable
2178 handlers. Subsequently, we will generate landing pads which will
2179 connect many of the handlers, and then type information will not
2180 be effective. Still, this is a win over previous implementations. */
2182 cleanup_cfg (CLEANUP_PRE_LOOP
| CLEANUP_NO_INSN_DEL
);
2184 /* These registers are used by the landing pads. Make sure they
2185 have been generated. */
2186 get_exception_pointer (cfun
);
2187 get_exception_filter (cfun
);
2189 /* Construct the landing pads. */
2191 assign_filter_values ();
2192 build_post_landing_pads ();
2193 connect_post_landing_pads ();
2194 if (USING_SJLJ_EXCEPTIONS
)
2195 sjlj_build_landing_pads ();
2197 dw2_build_landing_pads ();
2199 cfun
->eh
->built_landing_pads
= 1;
2201 /* We've totally changed the CFG. Start over. */
2202 find_exception_handler_labels ();
2203 break_superblocks ();
2204 if (USING_SJLJ_EXCEPTIONS
)
2205 commit_edge_insertions ();
2210 for (e
= bb
->succ
; e
; e
= next
)
2212 next
= e
->succ_next
;
2213 if (e
->flags
& EDGE_EH
)
2220 rtl_make_eh_edge (NULL
, bb
, BB_END (bb
));
2222 cleanup_cfg (CLEANUP_PRE_LOOP
| CLEANUP_NO_INSN_DEL
);
2226 ehl_hash (const void *pentry
)
2228 struct ehl_map_entry
*entry
= (struct ehl_map_entry
*) pentry
;
2230 /* 2^32 * ((sqrt(5) - 1) / 2) */
2231 const hashval_t scaled_golden_ratio
= 0x9e3779b9;
2232 return CODE_LABEL_NUMBER (entry
->label
) * scaled_golden_ratio
;
2236 ehl_eq (const void *pentry
, const void *pdata
)
2238 struct ehl_map_entry
*entry
= (struct ehl_map_entry
*) pentry
;
2239 struct ehl_map_entry
*data
= (struct ehl_map_entry
*) pdata
;
2241 return entry
->label
== data
->label
;
2244 /* This section handles removing dead code for flow. */
2246 /* Remove LABEL from exception_handler_label_map. */
2249 remove_exception_handler_label (rtx label
)
2251 struct ehl_map_entry
**slot
, tmp
;
2253 /* If exception_handler_label_map was not built yet,
2254 there is nothing to do. */
2255 if (cfun
->eh
->exception_handler_label_map
== NULL
)
2259 slot
= (struct ehl_map_entry
**)
2260 htab_find_slot (cfun
->eh
->exception_handler_label_map
, &tmp
, NO_INSERT
);
2264 htab_clear_slot (cfun
->eh
->exception_handler_label_map
, (void **) slot
);
2267 /* Splice REGION from the region tree etc. */
2270 remove_eh_handler (struct eh_region
*region
)
2272 struct eh_region
**pp
, **pp_start
, *p
, *outer
, *inner
;
2275 /* For the benefit of efficiently handling REG_EH_REGION notes,
2276 replace this region in the region array with its containing
2277 region. Note that previous region deletions may result in
2278 multiple copies of this region in the array, so we have a
2279 list of alternate numbers by which we are known. */
2281 outer
= region
->outer
;
2282 cfun
->eh
->region_array
[region
->region_number
] = outer
;
2286 EXECUTE_IF_SET_IN_BITMAP (region
->aka
, 0, i
,
2287 { cfun
->eh
->region_array
[i
] = outer
; });
2293 outer
->aka
= BITMAP_GGC_ALLOC ();
2295 bitmap_a_or_b (outer
->aka
, outer
->aka
, region
->aka
);
2296 bitmap_set_bit (outer
->aka
, region
->region_number
);
2299 if (cfun
->eh
->built_landing_pads
)
2300 lab
= region
->landing_pad
;
2302 lab
= region
->label
;
2304 remove_exception_handler_label (lab
);
2307 pp_start
= &outer
->inner
;
2309 pp_start
= &cfun
->eh
->region_tree
;
2310 for (pp
= pp_start
, p
= *pp
; p
!= region
; pp
= &p
->next_peer
, p
= *pp
)
2312 *pp
= region
->next_peer
;
2314 inner
= region
->inner
;
2317 for (p
= inner
; p
->next_peer
; p
= p
->next_peer
)
2321 p
->next_peer
= *pp_start
;
2325 if (region
->type
== ERT_CATCH
)
2327 struct eh_region
*try, *next
, *prev
;
2329 for (try = region
->next_peer
;
2330 try->type
== ERT_CATCH
;
2331 try = try->next_peer
)
2333 if (try->type
!= ERT_TRY
)
2336 next
= region
->u
.catch.next_catch
;
2337 prev
= region
->u
.catch.prev_catch
;
2340 next
->u
.catch.prev_catch
= prev
;
2342 try->u
.try.last_catch
= prev
;
2344 prev
->u
.catch.next_catch
= next
;
2347 try->u
.try.catch = next
;
2349 remove_eh_handler (try);
2354 /* LABEL heads a basic block that is about to be deleted. If this
2355 label corresponds to an exception region, we may be able to
2356 delete the region. */
2359 maybe_remove_eh_handler (rtx label
)
2361 struct ehl_map_entry
**slot
, tmp
;
2362 struct eh_region
*region
;
2364 /* ??? After generating landing pads, it's not so simple to determine
2365 if the region data is completely unused. One must examine the
2366 landing pad and the post landing pad, and whether an inner try block
2367 is referencing the catch handlers directly. */
2368 if (cfun
->eh
->built_landing_pads
)
2372 slot
= (struct ehl_map_entry
**)
2373 htab_find_slot (cfun
->eh
->exception_handler_label_map
, &tmp
, NO_INSERT
);
2376 region
= (*slot
)->region
;
2380 /* Flow will want to remove MUST_NOT_THROW regions as unreachable
2381 because there is no path to the fallback call to terminate.
2382 But the region continues to affect call-site data until there
2383 are no more contained calls, which we don't see here. */
2384 if (region
->type
== ERT_MUST_NOT_THROW
)
2386 htab_clear_slot (cfun
->eh
->exception_handler_label_map
, (void **) slot
);
2387 region
->label
= NULL_RTX
;
2390 remove_eh_handler (region
);
2393 /* Invokes CALLBACK for every exception handler label. Only used by old
2394 loop hackery; should not be used by new code. */
2397 for_each_eh_label (void (*callback
) (rtx
))
2399 htab_traverse (cfun
->eh
->exception_handler_label_map
, for_each_eh_label_1
,
2400 (void *) &callback
);
2404 for_each_eh_label_1 (void **pentry
, void *data
)
2406 struct ehl_map_entry
*entry
= *(struct ehl_map_entry
**)pentry
;
2407 void (*callback
) (rtx
) = *(void (**) (rtx
)) data
;
2409 (*callback
) (entry
->label
);
2413 /* Invoke CALLBACK for every exception region in the current function. */
2416 for_each_eh_region (void (*callback
) (struct eh_region
*))
2418 int i
, n
= cfun
->eh
->last_region_number
;
2419 for (i
= 1; i
<= n
; ++i
)
2421 struct eh_region
*region
= cfun
->eh
->region_array
[i
];
2423 (*callback
) (region
);
2427 /* This section describes CFG exception edges for flow. */
2429 /* For communicating between calls to reachable_next_level. */
2430 struct reachable_info
2434 void (*callback
) (struct eh_region
*, void *);
2435 void *callback_data
;
2436 bool saw_any_handlers
;
2439 /* A subroutine of reachable_next_level. Return true if TYPE, or a
2440 base class of TYPE, is in HANDLED. */
2443 check_handled (tree handled
, tree type
)
2447 /* We can check for exact matches without front-end help. */
2448 if (! lang_eh_type_covers
)
2450 for (t
= handled
; t
; t
= TREE_CHAIN (t
))
2451 if (TREE_VALUE (t
) == type
)
2456 for (t
= handled
; t
; t
= TREE_CHAIN (t
))
2457 if ((*lang_eh_type_covers
) (TREE_VALUE (t
), type
))
2464 /* A subroutine of reachable_next_level. If we are collecting a list
2465 of handlers, add one. After landing pad generation, reference
2466 it instead of the handlers themselves. Further, the handlers are
2467 all wired together, so by referencing one, we've got them all.
2468 Before landing pad generation we reference each handler individually.
2470 LP_REGION contains the landing pad; REGION is the handler. */
2473 add_reachable_handler (struct reachable_info
*info
,
2474 struct eh_region
*lp_region
, struct eh_region
*region
)
2479 info
->saw_any_handlers
= true;
2481 if (cfun
->eh
->built_landing_pads
)
2482 info
->callback (lp_region
, info
->callback_data
);
2484 info
->callback (region
, info
->callback_data
);
2487 /* Process one level of exception regions for reachability.
2488 If TYPE_THROWN is non-null, then it is the *exact* type being
2489 propagated. If INFO is non-null, then collect handler labels
2490 and caught/allowed type information between invocations. */
2492 static enum reachable_code
2493 reachable_next_level (struct eh_region
*region
, tree type_thrown
,
2494 struct reachable_info
*info
)
2496 switch (region
->type
)
2499 /* Before landing-pad generation, we model control flow
2500 directly to the individual handlers. In this way we can
2501 see that catch handler types may shadow one another. */
2502 add_reachable_handler (info
, region
, region
);
2503 return RNL_MAYBE_CAUGHT
;
2507 struct eh_region
*c
;
2508 enum reachable_code ret
= RNL_NOT_CAUGHT
;
2510 for (c
= region
->u
.try.catch; c
; c
= c
->u
.catch.next_catch
)
2512 /* A catch-all handler ends the search. */
2513 if (c
->u
.catch.type_list
== NULL
)
2515 add_reachable_handler (info
, region
, c
);
2521 /* If we have at least one type match, end the search. */
2522 tree tp_node
= c
->u
.catch.type_list
;
2524 for (; tp_node
; tp_node
= TREE_CHAIN (tp_node
))
2526 tree type
= TREE_VALUE (tp_node
);
2528 if (type
== type_thrown
2529 || (lang_eh_type_covers
2530 && (*lang_eh_type_covers
) (type
, type_thrown
)))
2532 add_reachable_handler (info
, region
, c
);
2537 /* If we have definitive information of a match failure,
2538 the catch won't trigger. */
2539 if (lang_eh_type_covers
)
2540 return RNL_NOT_CAUGHT
;
2543 /* At this point, we either don't know what type is thrown or
2544 don't have front-end assistance to help deciding if it is
2545 covered by one of the types in the list for this region.
2547 We'd then like to add this region to the list of reachable
2548 handlers since it is indeed potentially reachable based on the
2549 information we have.
2551 Actually, this handler is for sure not reachable if all the
2552 types it matches have already been caught. That is, it is only
2553 potentially reachable if at least one of the types it catches
2554 has not been previously caught. */
2557 ret
= RNL_MAYBE_CAUGHT
;
2560 tree tp_node
= c
->u
.catch.type_list
;
2561 bool maybe_reachable
= false;
2563 /* Compute the potential reachability of this handler and
2564 update the list of types caught at the same time. */
2565 for (; tp_node
; tp_node
= TREE_CHAIN (tp_node
))
2567 tree type
= TREE_VALUE (tp_node
);
2569 if (! check_handled (info
->types_caught
, type
))
2572 = tree_cons (NULL
, type
, info
->types_caught
);
2574 maybe_reachable
= true;
2578 if (maybe_reachable
)
2580 add_reachable_handler (info
, region
, c
);
2582 /* ??? If the catch type is a base class of every allowed
2583 type, then we know we can stop the search. */
2584 ret
= RNL_MAYBE_CAUGHT
;
2592 case ERT_ALLOWED_EXCEPTIONS
:
2593 /* An empty list of types definitely ends the search. */
2594 if (region
->u
.allowed
.type_list
== NULL_TREE
)
2596 add_reachable_handler (info
, region
, region
);
2600 /* Collect a list of lists of allowed types for use in detecting
2601 when a catch may be transformed into a catch-all. */
2603 info
->types_allowed
= tree_cons (NULL_TREE
,
2604 region
->u
.allowed
.type_list
,
2605 info
->types_allowed
);
2607 /* If we have definitive information about the type hierarchy,
2608 then we can tell if the thrown type will pass through the
2610 if (type_thrown
&& lang_eh_type_covers
)
2612 if (check_handled (region
->u
.allowed
.type_list
, type_thrown
))
2613 return RNL_NOT_CAUGHT
;
2616 add_reachable_handler (info
, region
, region
);
2621 add_reachable_handler (info
, region
, region
);
2622 return RNL_MAYBE_CAUGHT
;
2625 /* Catch regions are handled by their controlling try region. */
2626 return RNL_NOT_CAUGHT
;
2628 case ERT_MUST_NOT_THROW
:
2629 /* Here we end our search, since no exceptions may propagate.
2630 If we've touched down at some landing pad previous, then the
2631 explicit function call we generated may be used. Otherwise
2632 the call is made by the runtime. */
2633 if (info
&& info
->saw_any_handlers
)
2635 add_reachable_handler (info
, region
, region
);
2644 /* Shouldn't see these here. */
2651 /* Invoke CALLBACK on each region reachable from REGION_NUMBER. */
2654 foreach_reachable_handler (int region_number
, bool is_resx
,
2655 void (*callback
) (struct eh_region
*, void *),
2656 void *callback_data
)
2658 struct reachable_info info
;
2659 struct eh_region
*region
;
2662 memset (&info
, 0, sizeof (info
));
2663 info
.callback
= callback
;
2664 info
.callback_data
= callback_data
;
2666 region
= cfun
->eh
->region_array
[region_number
];
2668 type_thrown
= NULL_TREE
;
2671 /* A RESX leaves a region instead of entering it. Thus the
2672 region itself may have been deleted out from under us. */
2675 region
= region
->outer
;
2677 else if (region
->type
== ERT_THROW
)
2679 type_thrown
= region
->u
.throw.type
;
2680 region
= region
->outer
;
2685 if (reachable_next_level (region
, type_thrown
, &info
) >= RNL_CAUGHT
)
2687 /* If we have processed one cleanup, there is no point in
2688 processing any more of them. Each cleanup will have an edge
2689 to the next outer cleanup region, so the flow graph will be
2691 if (region
->type
== ERT_CLEANUP
)
2692 region
= region
->u
.cleanup
.prev_try
;
2694 region
= region
->outer
;
2698 /* Retrieve a list of labels of exception handlers which can be
2699 reached by a given insn. */
2702 arh_to_landing_pad (struct eh_region
*region
, void *data
)
2704 rtx
*p_handlers
= data
;
2706 *p_handlers
= alloc_INSN_LIST (region
->landing_pad
, NULL_RTX
);
2710 arh_to_label (struct eh_region
*region
, void *data
)
2712 rtx
*p_handlers
= data
;
2713 *p_handlers
= alloc_INSN_LIST (region
->label
, *p_handlers
);
2717 reachable_handlers (rtx insn
)
2719 bool is_resx
= false;
2720 rtx handlers
= NULL
;
2724 && GET_CODE (PATTERN (insn
)) == RESX
)
2726 region_number
= XINT (PATTERN (insn
), 0);
2731 rtx note
= find_reg_note (insn
, REG_EH_REGION
, NULL_RTX
);
2732 if (!note
|| INTVAL (XEXP (note
, 0)) <= 0)
2734 region_number
= INTVAL (XEXP (note
, 0));
2737 foreach_reachable_handler (region_number
, is_resx
,
2738 (cfun
->eh
->built_landing_pads
2739 ? arh_to_landing_pad
2746 /* Determine if the given INSN can throw an exception that is caught
2747 within the function. */
2750 can_throw_internal_1 (int region_number
)
2752 struct eh_region
*region
;
2755 region
= cfun
->eh
->region_array
[region_number
];
2757 type_thrown
= NULL_TREE
;
2758 if (region
->type
== ERT_THROW
)
2760 type_thrown
= region
->u
.throw.type
;
2761 region
= region
->outer
;
2764 /* If this exception is ignored by each and every containing region,
2765 then control passes straight out. The runtime may handle some
2766 regions, which also do not require processing internally. */
2767 for (; region
; region
= region
->outer
)
2769 enum reachable_code how
= reachable_next_level (region
, type_thrown
, 0);
2770 if (how
== RNL_BLOCKED
)
2772 if (how
!= RNL_NOT_CAUGHT
)
2780 can_throw_internal (rtx insn
)
2784 if (! INSN_P (insn
))
2788 && GET_CODE (PATTERN (insn
)) == RESX
2789 && XINT (PATTERN (insn
), 0) > 0)
2790 return can_throw_internal_1 (XINT (PATTERN (insn
), 0));
2792 if (NONJUMP_INSN_P (insn
)
2793 && GET_CODE (PATTERN (insn
)) == SEQUENCE
)
2794 insn
= XVECEXP (PATTERN (insn
), 0, 0);
2796 /* Every insn that might throw has an EH_REGION note. */
2797 note
= find_reg_note (insn
, REG_EH_REGION
, NULL_RTX
);
2798 if (!note
|| INTVAL (XEXP (note
, 0)) <= 0)
2801 return can_throw_internal_1 (INTVAL (XEXP (note
, 0)));
2804 /* Determine if the given INSN can throw an exception that is
2805 visible outside the function. */
2808 can_throw_external_1 (int region_number
)
2810 struct eh_region
*region
;
2813 region
= cfun
->eh
->region_array
[region_number
];
2815 type_thrown
= NULL_TREE
;
2816 if (region
->type
== ERT_THROW
)
2818 type_thrown
= region
->u
.throw.type
;
2819 region
= region
->outer
;
2822 /* If the exception is caught or blocked by any containing region,
2823 then it is not seen by any calling function. */
2824 for (; region
; region
= region
->outer
)
2825 if (reachable_next_level (region
, type_thrown
, NULL
) >= RNL_CAUGHT
)
2832 can_throw_external (rtx insn
)
2836 if (! INSN_P (insn
))
2839 if (NONJUMP_INSN_P (insn
)
2840 && GET_CODE (PATTERN (insn
)) == SEQUENCE
)
2841 insn
= XVECEXP (PATTERN (insn
), 0, 0);
2843 note
= find_reg_note (insn
, REG_EH_REGION
, NULL_RTX
);
2846 /* Calls (and trapping insns) without notes are outside any
2847 exception handling region in this function. We have to
2848 assume it might throw. Given that the front end and middle
2849 ends mark known NOTHROW functions, this isn't so wildly
2851 return (CALL_P (insn
)
2852 || (flag_non_call_exceptions
2853 && may_trap_p (PATTERN (insn
))));
2855 if (INTVAL (XEXP (note
, 0)) <= 0)
2858 return can_throw_external_1 (INTVAL (XEXP (note
, 0)));
2861 /* Set TREE_NOTHROW and cfun->all_throwers_are_sibcalls. */
2864 set_nothrow_function_flags (void)
2868 TREE_NOTHROW (current_function_decl
) = 1;
2870 /* Assume cfun->all_throwers_are_sibcalls until we encounter
2871 something that can throw an exception. We specifically exempt
2872 CALL_INSNs that are SIBLING_CALL_P, as these are really jumps,
2873 and can't throw. Most CALL_INSNs are not SIBLING_CALL_P, so this
2876 cfun
->all_throwers_are_sibcalls
= 1;
2878 if (! flag_exceptions
)
2881 for (insn
= get_insns (); insn
; insn
= NEXT_INSN (insn
))
2882 if (can_throw_external (insn
))
2884 TREE_NOTHROW (current_function_decl
) = 0;
2886 if (!CALL_P (insn
) || !SIBLING_CALL_P (insn
))
2888 cfun
->all_throwers_are_sibcalls
= 0;
2893 for (insn
= current_function_epilogue_delay_list
; insn
;
2894 insn
= XEXP (insn
, 1))
2895 if (can_throw_external (insn
))
2897 TREE_NOTHROW (current_function_decl
) = 0;
2899 if (!CALL_P (insn
) || !SIBLING_CALL_P (insn
))
2901 cfun
->all_throwers_are_sibcalls
= 0;
2908 /* Various hooks for unwind library. */
2910 /* Do any necessary initialization to access arbitrary stack frames.
2911 On the SPARC, this means flushing the register windows. */
2914 expand_builtin_unwind_init (void)
2916 /* Set this so all the registers get saved in our frame; we need to be
2917 able to copy the saved values for any registers from frames we unwind. */
2918 current_function_has_nonlocal_label
= 1;
2920 #ifdef SETUP_FRAME_ADDRESSES
2921 SETUP_FRAME_ADDRESSES ();
2926 expand_builtin_eh_return_data_regno (tree arglist
)
2928 tree which
= TREE_VALUE (arglist
);
2929 unsigned HOST_WIDE_INT iwhich
;
2931 if (TREE_CODE (which
) != INTEGER_CST
)
2933 error ("argument of `__builtin_eh_return_regno' must be constant");
2937 iwhich
= tree_low_cst (which
, 1);
2938 iwhich
= EH_RETURN_DATA_REGNO (iwhich
);
2939 if (iwhich
== INVALID_REGNUM
)
2942 #ifdef DWARF_FRAME_REGNUM
2943 iwhich
= DWARF_FRAME_REGNUM (iwhich
);
2945 iwhich
= DBX_REGISTER_NUMBER (iwhich
);
2948 return GEN_INT (iwhich
);
2951 /* Given a value extracted from the return address register or stack slot,
2952 return the actual address encoded in that value. */
2955 expand_builtin_extract_return_addr (tree addr_tree
)
2957 rtx addr
= expand_expr (addr_tree
, NULL_RTX
, Pmode
, 0);
2959 if (GET_MODE (addr
) != Pmode
2960 && GET_MODE (addr
) != VOIDmode
)
2962 #ifdef POINTERS_EXTEND_UNSIGNED
2963 addr
= convert_memory_address (Pmode
, addr
);
2965 addr
= convert_to_mode (Pmode
, addr
, 0);
2969 /* First mask out any unwanted bits. */
2970 #ifdef MASK_RETURN_ADDR
2971 expand_and (Pmode
, addr
, MASK_RETURN_ADDR
, addr
);
2974 /* Then adjust to find the real return address. */
2975 #if defined (RETURN_ADDR_OFFSET)
2976 addr
= plus_constant (addr
, RETURN_ADDR_OFFSET
);
2982 /* Given an actual address in addr_tree, do any necessary encoding
2983 and return the value to be stored in the return address register or
2984 stack slot so the epilogue will return to that address. */
2987 expand_builtin_frob_return_addr (tree addr_tree
)
2989 rtx addr
= expand_expr (addr_tree
, NULL_RTX
, ptr_mode
, 0);
2991 addr
= convert_memory_address (Pmode
, addr
);
2993 #ifdef RETURN_ADDR_OFFSET
2994 addr
= force_reg (Pmode
, addr
);
2995 addr
= plus_constant (addr
, -RETURN_ADDR_OFFSET
);
3001 /* Set up the epilogue with the magic bits we'll need to return to the
3002 exception handler. */
3005 expand_builtin_eh_return (tree stackadj_tree ATTRIBUTE_UNUSED
,
3010 #ifdef EH_RETURN_STACKADJ_RTX
3011 tmp
= expand_expr (stackadj_tree
, cfun
->eh
->ehr_stackadj
, VOIDmode
, 0);
3012 tmp
= convert_memory_address (Pmode
, tmp
);
3013 if (!cfun
->eh
->ehr_stackadj
)
3014 cfun
->eh
->ehr_stackadj
= copy_to_reg (tmp
);
3015 else if (tmp
!= cfun
->eh
->ehr_stackadj
)
3016 emit_move_insn (cfun
->eh
->ehr_stackadj
, tmp
);
3019 tmp
= expand_expr (handler_tree
, cfun
->eh
->ehr_handler
, VOIDmode
, 0);
3020 tmp
= convert_memory_address (Pmode
, tmp
);
3021 if (!cfun
->eh
->ehr_handler
)
3022 cfun
->eh
->ehr_handler
= copy_to_reg (tmp
);
3023 else if (tmp
!= cfun
->eh
->ehr_handler
)
3024 emit_move_insn (cfun
->eh
->ehr_handler
, tmp
);
3026 if (!cfun
->eh
->ehr_label
)
3027 cfun
->eh
->ehr_label
= gen_label_rtx ();
3028 emit_jump (cfun
->eh
->ehr_label
);
3032 expand_eh_return (void)
3036 if (! cfun
->eh
->ehr_label
)
3039 current_function_calls_eh_return
= 1;
3041 #ifdef EH_RETURN_STACKADJ_RTX
3042 emit_move_insn (EH_RETURN_STACKADJ_RTX
, const0_rtx
);
3045 around_label
= gen_label_rtx ();
3046 emit_jump (around_label
);
3048 emit_label (cfun
->eh
->ehr_label
);
3049 clobber_return_register ();
3051 #ifdef EH_RETURN_STACKADJ_RTX
3052 emit_move_insn (EH_RETURN_STACKADJ_RTX
, cfun
->eh
->ehr_stackadj
);
3055 #ifdef HAVE_eh_return
3057 emit_insn (gen_eh_return (cfun
->eh
->ehr_handler
));
3061 #ifdef EH_RETURN_HANDLER_RTX
3062 emit_move_insn (EH_RETURN_HANDLER_RTX
, cfun
->eh
->ehr_handler
);
3064 error ("__builtin_eh_return not supported on this target");
3068 emit_label (around_label
);
3071 /* Convert a ptr_mode address ADDR_TREE to a Pmode address controlled by
3072 POINTERS_EXTEND_UNSIGNED and return it. */
3075 expand_builtin_extend_pointer (tree addr_tree
)
3077 rtx addr
= expand_expr (addr_tree
, NULL_RTX
, ptr_mode
, 0);
3080 #ifdef POINTERS_EXTEND_UNSIGNED
3081 extend
= POINTERS_EXTEND_UNSIGNED
;
3083 /* The previous EH code did an unsigned extend by default, so we do this also
3088 return convert_modes (word_mode
, ptr_mode
, addr
, extend
);
3091 /* In the following functions, we represent entries in the action table
3092 as 1-based indices. Special cases are:
3094 0: null action record, non-null landing pad; implies cleanups
3095 -1: null action record, null landing pad; implies no action
3096 -2: no call-site entry; implies must_not_throw
3097 -3: we have yet to process outer regions
3099 Further, no special cases apply to the "next" field of the record.
3100 For next, 0 means end of list. */
3102 struct action_record
3110 action_record_eq (const void *pentry
, const void *pdata
)
3112 const struct action_record
*entry
= (const struct action_record
*) pentry
;
3113 const struct action_record
*data
= (const struct action_record
*) pdata
;
3114 return entry
->filter
== data
->filter
&& entry
->next
== data
->next
;
3118 action_record_hash (const void *pentry
)
3120 const struct action_record
*entry
= (const struct action_record
*) pentry
;
3121 return entry
->next
* 1009 + entry
->filter
;
3125 add_action_record (htab_t ar_hash
, int filter
, int next
)
3127 struct action_record
**slot
, *new, tmp
;
3129 tmp
.filter
= filter
;
3131 slot
= (struct action_record
**) htab_find_slot (ar_hash
, &tmp
, INSERT
);
3133 if ((new = *slot
) == NULL
)
3135 new = xmalloc (sizeof (*new));
3136 new->offset
= VARRAY_ACTIVE_SIZE (cfun
->eh
->action_record_data
) + 1;
3137 new->filter
= filter
;
3141 /* The filter value goes in untouched. The link to the next
3142 record is a "self-relative" byte offset, or zero to indicate
3143 that there is no next record. So convert the absolute 1 based
3144 indices we've been carrying around into a displacement. */
3146 push_sleb128 (&cfun
->eh
->action_record_data
, filter
);
3148 next
-= VARRAY_ACTIVE_SIZE (cfun
->eh
->action_record_data
) + 1;
3149 push_sleb128 (&cfun
->eh
->action_record_data
, next
);
3156 collect_one_action_chain (htab_t ar_hash
, struct eh_region
*region
)
3158 struct eh_region
*c
;
3161 /* If we've reached the top of the region chain, then we have
3162 no actions, and require no landing pad. */
3166 switch (region
->type
)
3169 /* A cleanup adds a zero filter to the beginning of the chain, but
3170 there are special cases to look out for. If there are *only*
3171 cleanups along a path, then it compresses to a zero action.
3172 Further, if there are multiple cleanups along a path, we only
3173 need to represent one of them, as that is enough to trigger
3174 entry to the landing pad at runtime. */
3175 next
= collect_one_action_chain (ar_hash
, region
->outer
);
3178 for (c
= region
->outer
; c
; c
= c
->outer
)
3179 if (c
->type
== ERT_CLEANUP
)
3181 return add_action_record (ar_hash
, 0, next
);
3184 /* Process the associated catch regions in reverse order.
3185 If there's a catch-all handler, then we don't need to
3186 search outer regions. Use a magic -3 value to record
3187 that we haven't done the outer search. */
3189 for (c
= region
->u
.try.last_catch
; c
; c
= c
->u
.catch.prev_catch
)
3191 if (c
->u
.catch.type_list
== NULL
)
3193 /* Retrieve the filter from the head of the filter list
3194 where we have stored it (see assign_filter_values). */
3196 = TREE_INT_CST_LOW (TREE_VALUE (c
->u
.catch.filter_list
));
3198 next
= add_action_record (ar_hash
, filter
, 0);
3202 /* Once the outer search is done, trigger an action record for
3203 each filter we have. */
3208 next
= collect_one_action_chain (ar_hash
, region
->outer
);
3210 /* If there is no next action, terminate the chain. */
3213 /* If all outer actions are cleanups or must_not_throw,
3214 we'll have no action record for it, since we had wanted
3215 to encode these states in the call-site record directly.
3216 Add a cleanup action to the chain to catch these. */
3218 next
= add_action_record (ar_hash
, 0, 0);
3221 flt_node
= c
->u
.catch.filter_list
;
3222 for (; flt_node
; flt_node
= TREE_CHAIN (flt_node
))
3224 int filter
= TREE_INT_CST_LOW (TREE_VALUE (flt_node
));
3225 next
= add_action_record (ar_hash
, filter
, next
);
3231 case ERT_ALLOWED_EXCEPTIONS
:
3232 /* An exception specification adds its filter to the
3233 beginning of the chain. */
3234 next
= collect_one_action_chain (ar_hash
, region
->outer
);
3236 /* If there is no next action, terminate the chain. */
3239 /* If all outer actions are cleanups or must_not_throw,
3240 we'll have no action record for it, since we had wanted
3241 to encode these states in the call-site record directly.
3242 Add a cleanup action to the chain to catch these. */
3244 next
= add_action_record (ar_hash
, 0, 0);
3246 return add_action_record (ar_hash
, region
->u
.allowed
.filter
, next
);
3248 case ERT_MUST_NOT_THROW
:
3249 /* A must-not-throw region with no inner handlers or cleanups
3250 requires no call-site entry. Note that this differs from
3251 the no handler or cleanup case in that we do require an lsda
3252 to be generated. Return a magic -2 value to record this. */
3257 /* CATCH regions are handled in TRY above. THROW regions are
3258 for optimization information only and produce no output. */
3259 return collect_one_action_chain (ar_hash
, region
->outer
);
3267 add_call_site (rtx landing_pad
, int action
)
3269 struct call_site_record
*data
= cfun
->eh
->call_site_data
;
3270 int used
= cfun
->eh
->call_site_data_used
;
3271 int size
= cfun
->eh
->call_site_data_size
;
3275 size
= (size
? size
* 2 : 64);
3276 data
= ggc_realloc (data
, sizeof (*data
) * size
);
3277 cfun
->eh
->call_site_data
= data
;
3278 cfun
->eh
->call_site_data_size
= size
;
3281 data
[used
].landing_pad
= landing_pad
;
3282 data
[used
].action
= action
;
3284 cfun
->eh
->call_site_data_used
= used
+ 1;
3286 return used
+ call_site_base
;
3289 /* Turn REG_EH_REGION notes back into NOTE_INSN_EH_REGION notes.
3290 The new note numbers will not refer to region numbers, but
3291 instead to call site entries. */
3294 convert_to_eh_region_ranges (void)
3296 rtx insn
, iter
, note
;
3298 int last_action
= -3;
3299 rtx last_action_insn
= NULL_RTX
;
3300 rtx last_landing_pad
= NULL_RTX
;
3301 rtx first_no_action_insn
= NULL_RTX
;
3304 if (USING_SJLJ_EXCEPTIONS
|| cfun
->eh
->region_tree
== NULL
)
3307 VARRAY_UCHAR_INIT (cfun
->eh
->action_record_data
, 64, "action_record_data");
3309 ar_hash
= htab_create (31, action_record_hash
, action_record_eq
, free
);
3311 for (iter
= get_insns (); iter
; iter
= NEXT_INSN (iter
))
3314 struct eh_region
*region
;
3316 rtx this_landing_pad
;
3319 if (NONJUMP_INSN_P (insn
)
3320 && GET_CODE (PATTERN (insn
)) == SEQUENCE
)
3321 insn
= XVECEXP (PATTERN (insn
), 0, 0);
3323 note
= find_reg_note (insn
, REG_EH_REGION
, NULL_RTX
);
3326 if (! (CALL_P (insn
)
3327 || (flag_non_call_exceptions
3328 && may_trap_p (PATTERN (insn
)))))
3335 if (INTVAL (XEXP (note
, 0)) <= 0)
3337 region
= cfun
->eh
->region_array
[INTVAL (XEXP (note
, 0))];
3338 this_action
= collect_one_action_chain (ar_hash
, region
);
3341 /* Existence of catch handlers, or must-not-throw regions
3342 implies that an lsda is needed (even if empty). */
3343 if (this_action
!= -1)
3344 cfun
->uses_eh_lsda
= 1;
3346 /* Delay creation of region notes for no-action regions
3347 until we're sure that an lsda will be required. */
3348 else if (last_action
== -3)
3350 first_no_action_insn
= iter
;
3354 /* Cleanups and handlers may share action chains but not
3355 landing pads. Collect the landing pad for this region. */
3356 if (this_action
>= 0)
3358 struct eh_region
*o
;
3359 for (o
= region
; ! o
->landing_pad
; o
= o
->outer
)
3361 this_landing_pad
= o
->landing_pad
;
3364 this_landing_pad
= NULL_RTX
;
3366 /* Differing actions or landing pads implies a change in call-site
3367 info, which implies some EH_REGION note should be emitted. */
3368 if (last_action
!= this_action
3369 || last_landing_pad
!= this_landing_pad
)
3371 /* If we'd not seen a previous action (-3) or the previous
3372 action was must-not-throw (-2), then we do not need an
3374 if (last_action
>= -1)
3376 /* If we delayed the creation of the begin, do it now. */
3377 if (first_no_action_insn
)
3379 call_site
= add_call_site (NULL_RTX
, 0);
3380 note
= emit_note_before (NOTE_INSN_EH_REGION_BEG
,
3381 first_no_action_insn
);
3382 NOTE_EH_HANDLER (note
) = call_site
;
3383 first_no_action_insn
= NULL_RTX
;
3386 note
= emit_note_after (NOTE_INSN_EH_REGION_END
,
3388 NOTE_EH_HANDLER (note
) = call_site
;
3391 /* If the new action is must-not-throw, then no region notes
3393 if (this_action
>= -1)
3395 call_site
= add_call_site (this_landing_pad
,
3396 this_action
< 0 ? 0 : this_action
);
3397 note
= emit_note_before (NOTE_INSN_EH_REGION_BEG
, iter
);
3398 NOTE_EH_HANDLER (note
) = call_site
;
3401 last_action
= this_action
;
3402 last_landing_pad
= this_landing_pad
;
3404 last_action_insn
= iter
;
3407 if (last_action
>= -1 && ! first_no_action_insn
)
3409 note
= emit_note_after (NOTE_INSN_EH_REGION_END
, last_action_insn
);
3410 NOTE_EH_HANDLER (note
) = call_site
;
3413 htab_delete (ar_hash
);
3418 push_uleb128 (varray_type
*data_area
, unsigned int value
)
3422 unsigned char byte
= value
& 0x7f;
3426 VARRAY_PUSH_UCHAR (*data_area
, byte
);
3432 push_sleb128 (varray_type
*data_area
, int value
)
3439 byte
= value
& 0x7f;
3441 more
= ! ((value
== 0 && (byte
& 0x40) == 0)
3442 || (value
== -1 && (byte
& 0x40) != 0));
3445 VARRAY_PUSH_UCHAR (*data_area
, byte
);
3451 #ifndef HAVE_AS_LEB128
3453 dw2_size_of_call_site_table (void)
3455 int n
= cfun
->eh
->call_site_data_used
;
3456 int size
= n
* (4 + 4 + 4);
3459 for (i
= 0; i
< n
; ++i
)
3461 struct call_site_record
*cs
= &cfun
->eh
->call_site_data
[i
];
3462 size
+= size_of_uleb128 (cs
->action
);
3469 sjlj_size_of_call_site_table (void)
3471 int n
= cfun
->eh
->call_site_data_used
;
3475 for (i
= 0; i
< n
; ++i
)
3477 struct call_site_record
*cs
= &cfun
->eh
->call_site_data
[i
];
3478 size
+= size_of_uleb128 (INTVAL (cs
->landing_pad
));
3479 size
+= size_of_uleb128 (cs
->action
);
3487 dw2_output_call_site_table (void)
3489 const char *const function_start_lab
3490 = IDENTIFIER_POINTER (current_function_func_begin_label
);
3491 int n
= cfun
->eh
->call_site_data_used
;
3494 for (i
= 0; i
< n
; ++i
)
3496 struct call_site_record
*cs
= &cfun
->eh
->call_site_data
[i
];
3497 char reg_start_lab
[32];
3498 char reg_end_lab
[32];
3499 char landing_pad_lab
[32];
3501 ASM_GENERATE_INTERNAL_LABEL (reg_start_lab
, "LEHB", call_site_base
+ i
);
3502 ASM_GENERATE_INTERNAL_LABEL (reg_end_lab
, "LEHE", call_site_base
+ i
);
3504 if (cs
->landing_pad
)
3505 ASM_GENERATE_INTERNAL_LABEL (landing_pad_lab
, "L",
3506 CODE_LABEL_NUMBER (cs
->landing_pad
));
3508 /* ??? Perhaps use insn length scaling if the assembler supports
3509 generic arithmetic. */
3510 /* ??? Perhaps use attr_length to choose data1 or data2 instead of
3511 data4 if the function is small enough. */
3512 #ifdef HAVE_AS_LEB128
3513 dw2_asm_output_delta_uleb128 (reg_start_lab
, function_start_lab
,
3514 "region %d start", i
);
3515 dw2_asm_output_delta_uleb128 (reg_end_lab
, reg_start_lab
,
3517 if (cs
->landing_pad
)
3518 dw2_asm_output_delta_uleb128 (landing_pad_lab
, function_start_lab
,
3521 dw2_asm_output_data_uleb128 (0, "landing pad");
3523 dw2_asm_output_delta (4, reg_start_lab
, function_start_lab
,
3524 "region %d start", i
);
3525 dw2_asm_output_delta (4, reg_end_lab
, reg_start_lab
, "length");
3526 if (cs
->landing_pad
)
3527 dw2_asm_output_delta (4, landing_pad_lab
, function_start_lab
,
3530 dw2_asm_output_data (4, 0, "landing pad");
3532 dw2_asm_output_data_uleb128 (cs
->action
, "action");
3535 call_site_base
+= n
;
3539 sjlj_output_call_site_table (void)
3541 int n
= cfun
->eh
->call_site_data_used
;
3544 for (i
= 0; i
< n
; ++i
)
3546 struct call_site_record
*cs
= &cfun
->eh
->call_site_data
[i
];
3548 dw2_asm_output_data_uleb128 (INTVAL (cs
->landing_pad
),
3549 "region %d landing pad", i
);
3550 dw2_asm_output_data_uleb128 (cs
->action
, "action");
3553 call_site_base
+= n
;
3556 /* Tell assembler to switch to the section for the exception handling
3560 default_exception_section (void)
3562 if (targetm
.have_named_sections
)
3565 #ifdef HAVE_LD_RO_RW_SECTION_MIXING
3566 int tt_format
= ASM_PREFERRED_EH_DATA_FORMAT (/*code=*/0, /*global=*/1);
3569 || ((tt_format
& 0x70) != DW_EH_PE_absptr
3570 && (tt_format
& 0x70) != DW_EH_PE_aligned
))
3571 ? 0 : SECTION_WRITE
;
3573 flags
= SECTION_WRITE
;
3575 named_section_flags (".gcc_except_table", flags
);
3580 readonly_data_section ();
3584 output_function_exception_table (void)
3586 int tt_format
, cs_format
, lp_format
, i
, n
;
3587 #ifdef HAVE_AS_LEB128
3588 char ttype_label
[32];
3589 char cs_after_size_label
[32];
3590 char cs_end_label
[32];
3595 int tt_format_size
= 0;
3597 /* Not all functions need anything. */
3598 if (! cfun
->uses_eh_lsda
)
3601 #ifdef IA64_UNWIND_INFO
3602 fputs ("\t.personality\t", asm_out_file
);
3603 output_addr_const (asm_out_file
, eh_personality_libfunc
);
3604 fputs ("\n\t.handlerdata\n", asm_out_file
);
3605 /* Note that varasm still thinks we're in the function's code section.
3606 The ".endp" directive that will immediately follow will take us back. */
3608 targetm
.asm_out
.exception_section ();
3611 have_tt_data
= (VARRAY_ACTIVE_SIZE (cfun
->eh
->ttype_data
) > 0
3612 || VARRAY_ACTIVE_SIZE (cfun
->eh
->ehspec_data
) > 0);
3614 /* Indicate the format of the @TType entries. */
3616 tt_format
= DW_EH_PE_omit
;
3619 tt_format
= ASM_PREFERRED_EH_DATA_FORMAT (/*code=*/0, /*global=*/1);
3620 #ifdef HAVE_AS_LEB128
3621 ASM_GENERATE_INTERNAL_LABEL (ttype_label
, "LLSDATT",
3622 current_function_funcdef_no
);
3624 tt_format_size
= size_of_encoded_value (tt_format
);
3626 assemble_align (tt_format_size
* BITS_PER_UNIT
);
3629 targetm
.asm_out
.internal_label (asm_out_file
, "LLSDA",
3630 current_function_funcdef_no
);
3632 /* The LSDA header. */
3634 /* Indicate the format of the landing pad start pointer. An omitted
3635 field implies @LPStart == @Start. */
3636 /* Currently we always put @LPStart == @Start. This field would
3637 be most useful in moving the landing pads completely out of
3638 line to another section, but it could also be used to minimize
3639 the size of uleb128 landing pad offsets. */
3640 lp_format
= DW_EH_PE_omit
;
3641 dw2_asm_output_data (1, lp_format
, "@LPStart format (%s)",
3642 eh_data_format_name (lp_format
));
3644 /* @LPStart pointer would go here. */
3646 dw2_asm_output_data (1, tt_format
, "@TType format (%s)",
3647 eh_data_format_name (tt_format
));
3649 #ifndef HAVE_AS_LEB128
3650 if (USING_SJLJ_EXCEPTIONS
)
3651 call_site_len
= sjlj_size_of_call_site_table ();
3653 call_site_len
= dw2_size_of_call_site_table ();
3656 /* A pc-relative 4-byte displacement to the @TType data. */
3659 #ifdef HAVE_AS_LEB128
3660 char ttype_after_disp_label
[32];
3661 ASM_GENERATE_INTERNAL_LABEL (ttype_after_disp_label
, "LLSDATTD",
3662 current_function_funcdef_no
);
3663 dw2_asm_output_delta_uleb128 (ttype_label
, ttype_after_disp_label
,
3664 "@TType base offset");
3665 ASM_OUTPUT_LABEL (asm_out_file
, ttype_after_disp_label
);
3667 /* Ug. Alignment queers things. */
3668 unsigned int before_disp
, after_disp
, last_disp
, disp
;
3670 before_disp
= 1 + 1;
3671 after_disp
= (1 + size_of_uleb128 (call_site_len
)
3673 + VARRAY_ACTIVE_SIZE (cfun
->eh
->action_record_data
)
3674 + (VARRAY_ACTIVE_SIZE (cfun
->eh
->ttype_data
)
3680 unsigned int disp_size
, pad
;
3683 disp_size
= size_of_uleb128 (disp
);
3684 pad
= before_disp
+ disp_size
+ after_disp
;
3685 if (pad
% tt_format_size
)
3686 pad
= tt_format_size
- (pad
% tt_format_size
);
3689 disp
= after_disp
+ pad
;
3691 while (disp
!= last_disp
);
3693 dw2_asm_output_data_uleb128 (disp
, "@TType base offset");
3697 /* Indicate the format of the call-site offsets. */
3698 #ifdef HAVE_AS_LEB128
3699 cs_format
= DW_EH_PE_uleb128
;
3701 cs_format
= DW_EH_PE_udata4
;
3703 dw2_asm_output_data (1, cs_format
, "call-site format (%s)",
3704 eh_data_format_name (cs_format
));
3706 #ifdef HAVE_AS_LEB128
3707 ASM_GENERATE_INTERNAL_LABEL (cs_after_size_label
, "LLSDACSB",
3708 current_function_funcdef_no
);
3709 ASM_GENERATE_INTERNAL_LABEL (cs_end_label
, "LLSDACSE",
3710 current_function_funcdef_no
);
3711 dw2_asm_output_delta_uleb128 (cs_end_label
, cs_after_size_label
,
3712 "Call-site table length");
3713 ASM_OUTPUT_LABEL (asm_out_file
, cs_after_size_label
);
3714 if (USING_SJLJ_EXCEPTIONS
)
3715 sjlj_output_call_site_table ();
3717 dw2_output_call_site_table ();
3718 ASM_OUTPUT_LABEL (asm_out_file
, cs_end_label
);
3720 dw2_asm_output_data_uleb128 (call_site_len
,"Call-site table length");
3721 if (USING_SJLJ_EXCEPTIONS
)
3722 sjlj_output_call_site_table ();
3724 dw2_output_call_site_table ();
3727 /* ??? Decode and interpret the data for flag_debug_asm. */
3728 n
= VARRAY_ACTIVE_SIZE (cfun
->eh
->action_record_data
);
3729 for (i
= 0; i
< n
; ++i
)
3730 dw2_asm_output_data (1, VARRAY_UCHAR (cfun
->eh
->action_record_data
, i
),
3731 (i
? NULL
: "Action record table"));
3734 assemble_align (tt_format_size
* BITS_PER_UNIT
);
3736 i
= VARRAY_ACTIVE_SIZE (cfun
->eh
->ttype_data
);
3739 tree type
= VARRAY_TREE (cfun
->eh
->ttype_data
, i
);
3742 if (type
== NULL_TREE
)
3746 struct cgraph_varpool_node
*node
;
3748 type
= lookup_type_for_runtime (type
);
3749 value
= expand_expr (type
, NULL_RTX
, VOIDmode
, EXPAND_INITIALIZER
);
3751 /* Let cgraph know that the rtti decl is used. Not all of the
3752 paths below go through assemble_integer, which would take
3753 care of this for us. */
3755 if (TREE_CODE (type
) == ADDR_EXPR
)
3757 type
= TREE_OPERAND (type
, 0);
3758 if (TREE_CODE (type
) == VAR_DECL
)
3760 node
= cgraph_varpool_node (type
);
3762 cgraph_varpool_mark_needed_node (node
);
3765 else if (TREE_CODE (type
) != INTEGER_CST
)
3769 if (tt_format
== DW_EH_PE_absptr
|| tt_format
== DW_EH_PE_aligned
)
3770 assemble_integer (value
, tt_format_size
,
3771 tt_format_size
* BITS_PER_UNIT
, 1);
3773 dw2_asm_output_encoded_addr_rtx (tt_format
, value
, NULL
);
3776 #ifdef HAVE_AS_LEB128
3778 ASM_OUTPUT_LABEL (asm_out_file
, ttype_label
);
3781 /* ??? Decode and interpret the data for flag_debug_asm. */
3782 n
= VARRAY_ACTIVE_SIZE (cfun
->eh
->ehspec_data
);
3783 for (i
= 0; i
< n
; ++i
)
3784 dw2_asm_output_data (1, VARRAY_UCHAR (cfun
->eh
->ehspec_data
, i
),
3785 (i
? NULL
: "Exception specification table"));
3787 function_section (current_function_decl
);
3790 #include "gt-except.h"