OSDN Git Service

* config/rs6000/darwin.h (SUBTARGET_OPTIONS): Move from here, to...
[pf3gnuchains/gcc-fork.git] / gcc / except.c
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>.
5
6 This file is part of GCC.
7
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
11 version.
12
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
16 for more details.
17
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
21 02111-1307, USA.  */
22
23
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.
29
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.
39
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.
46
47    [ Add updated documentation on how to use this.  ]  */
48
49
50 #include "config.h"
51 #include "system.h"
52 #include "coretypes.h"
53 #include "tm.h"
54 #include "rtl.h"
55 #include "tree.h"
56 #include "flags.h"
57 #include "function.h"
58 #include "expr.h"
59 #include "libfuncs.h"
60 #include "insn-config.h"
61 #include "except.h"
62 #include "integrate.h"
63 #include "hard-reg-set.h"
64 #include "basic-block.h"
65 #include "output.h"
66 #include "dwarf2asm.h"
67 #include "dwarf2out.h"
68 #include "dwarf2.h"
69 #include "toplev.h"
70 #include "hashtab.h"
71 #include "intl.h"
72 #include "ggc.h"
73 #include "tm_p.h"
74 #include "target.h"
75 #include "langhooks.h"
76 #include "cgraph.h"
77
78 /* Provide defaults for stuff that may not be defined when using
79    sjlj exceptions.  */
80 #ifndef EH_RETURN_DATA_REGNO
81 #define EH_RETURN_DATA_REGNO(N) INVALID_REGNUM
82 #endif
83
84
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);
88
89 /* Return true if type A catches type B.  */
90 int (*lang_eh_type_covers) (tree a, tree b);
91
92 /* Map a type to a runtime object to match type.  */
93 tree (*lang_eh_runtime_type) (tree);
94
95 /* A hash table of label to region number.  */
96
97 struct ehl_map_entry GTY(())
98 {
99   rtx label;
100   struct eh_region *region;
101 };
102
103 static GTY(()) int call_site_base;
104 static GTY ((param_is (union tree_node)))
105   htab_t type_to_runtime_map;
106
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;
114 \f
115 /* Describes one exception region.  */
116 struct eh_region GTY(())
117 {
118   /* The immediately surrounding region.  */
119   struct eh_region *outer;
120
121   /* The list of immediately contained regions.  */
122   struct eh_region *inner;
123   struct eh_region *next_peer;
124
125   /* An identifier for this region.  */
126   int region_number;
127
128   /* When a region is deleted, its parents inherit the REG_EH_REGION
129      numbers already assigned.  */
130   bitmap aka;
131
132   /* Each region does exactly one thing.  */
133   enum eh_region_type
134   { 
135     ERT_UNKNOWN = 0,
136     ERT_CLEANUP,
137     ERT_TRY,
138     ERT_CATCH,
139     ERT_ALLOWED_EXCEPTIONS,
140     ERT_MUST_NOT_THROW,
141     ERT_THROW,
142     ERT_FIXUP
143   } type;
144
145   /* Holds the action to perform based on the preceding type.  */
146   union eh_region_u {
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;
153       rtx continue_label;
154     } GTY ((tag ("ERT_TRY"))) try;
155
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;
161       tree type_list;
162       tree filter_list;
163     } GTY ((tag ("ERT_CATCH"))) catch;
164
165     /* A tree_list of allowed types.  */
166     struct eh_region_u_allowed {
167       tree type_list;
168       int filter;
169     } GTY ((tag ("ERT_ALLOWED_EXCEPTIONS"))) allowed;
170
171     /* The type given by a call to "throw foo();", or discovered
172        for a throw.  */
173     struct eh_region_u_throw {
174       tree type;
175     } GTY ((tag ("ERT_THROW"))) throw;
176
177     /* Retain the cleanup expression even after expansion so that
178        we can match up fixup regions.  */
179     struct eh_region_u_cleanup {
180       tree exp;
181       struct eh_region *prev_try;
182     } GTY ((tag ("ERT_CLEANUP"))) cleanup;
183
184     /* The real region (by expression and by pointer) that fixup code
185        should live in.  */
186     struct eh_region_u_fixup {
187       tree cleanup_exp;
188       struct eh_region *real_region;
189       bool resolved;
190     } GTY ((tag ("ERT_FIXUP"))) fixup;
191   } GTY ((desc ("%0.type"))) u;
192
193   /* Entry point for this region's handler before landing pads are built.  */
194   rtx label;
195   tree tree_label;
196
197   /* Entry point for this region's handler from the runtime eh library.  */
198   rtx landing_pad;
199
200   /* Entry point for this region's handler from an inner region.  */
201   rtx post_landing_pad;
202
203   /* The RESX insn for handing off control to the next outermost handler,
204      if appropriate.  */
205   rtx resume;
206
207   /* True if something in this region may throw.  */
208   unsigned may_contain_throw : 1;
209 };
210
211 struct call_site_record GTY(())
212 {
213   rtx landing_pad;
214   int action;
215 };
216
217 /* Used to save exception status for each function.  */
218 struct eh_status GTY(())
219 {
220   /* The tree of all regions for this function.  */
221   struct eh_region *region_tree;
222
223   /* The same information as an indexable array.  */
224   struct eh_region ** GTY ((length ("%h.last_region_number"))) region_array;
225
226   /* The most recently open region.  */
227   struct eh_region *cur_region;
228
229   /* This is the region for which we are processing catch blocks.  */
230   struct eh_region *try_region;
231
232   rtx filter;
233   rtx exc_ptr;
234
235   int built_landing_pads;
236   int last_region_number;
237
238   varray_type ttype_data;
239   varray_type ehspec_data;
240   varray_type action_record_data;
241
242   htab_t GTY ((param_is (struct ehl_map_entry))) exception_handler_label_map;
243
244   struct call_site_record * GTY ((length ("%h.call_site_data_used")))
245     call_site_data;
246   int call_site_data_used;
247   int call_site_data_size;
248
249   rtx ehr_stackadj;
250   rtx ehr_handler;
251   rtx ehr_label;
252
253   rtx sjlj_fc;
254   rtx sjlj_exit_after;
255 };
256
257 \f
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);
262
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);
267
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);
281
282 struct sjlj_lp_info;
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);
290
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 *);
297
298 /* The return value of reachable_next_level.  */
299 enum reachable_code
300 {
301   /* The given exception is not processed by the given region.  */
302   RNL_NOT_CAUGHT,
303   /* The given exception may need processing by the given region.  */
304   RNL_MAYBE_CAUGHT,
305   /* The given exception is completely processed by the given region.  */
306   RNL_CAUGHT,
307   /* The given exception is completely processed by the runtime.  */
308   RNL_BLOCKED
309 };
310
311 struct reachable_info;
312 static enum reachable_code reachable_next_level (struct eh_region *, tree,
313                                                  struct reachable_info *);
314
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);
320
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);
326 #endif
327 static void dw2_output_call_site_table (void);
328 static void sjlj_output_call_site_table (void);
329
330 \f
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.
334
335    This is used to ensure that -fexceptions has been specified if the
336    compiler tries to use any exception-specific functions.  */
337
338 int
339 doing_eh (int do_warn)
340 {
341   if (! flag_exceptions)
342     {
343       static int warned = 0;
344       if (! warned && do_warn)
345         {
346           error ("exception handling disabled, use -fexceptions to enable");
347           warned = 1;
348         }
349       return 0;
350     }
351   return 1;
352 }
353
354 \f
355 void
356 init_eh (void)
357 {
358   if (! flag_exceptions)
359     return;
360
361   type_to_runtime_map = htab_create_ggc (31, t2r_hash, t2r_eq, NULL);
362
363   /* Create the SjLj_Function_Context structure.  This should match
364      the definition in unwind-sjlj.c.  */
365   if (USING_SJLJ_EXCEPTIONS)
366     {
367       tree f_jbuf, f_per, f_lsda, f_prev, f_cs, f_data, tmp;
368
369       sjlj_fc_type_node = lang_hooks.types.make_type (RECORD_TYPE);
370
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;
374
375       f_cs = build_decl (FIELD_DECL, get_identifier ("__call_site"),
376                          integer_type_node);
377       DECL_FIELD_CONTEXT (f_cs) = sjlj_fc_type_node;
378
379       tmp = build_index_type (build_int_cst (NULL_TREE, 4 - 1, 0));
380       tmp = build_array_type (lang_hooks.types.type_for_mode (word_mode, 1),
381                               tmp);
382       f_data = build_decl (FIELD_DECL, get_identifier ("__data"), tmp);
383       DECL_FIELD_CONTEXT (f_data) = sjlj_fc_type_node;
384
385       f_per = build_decl (FIELD_DECL, get_identifier ("__personality"),
386                           ptr_type_node);
387       DECL_FIELD_CONTEXT (f_per) = sjlj_fc_type_node;
388
389       f_lsda = build_decl (FIELD_DECL, get_identifier ("__lsda"),
390                            ptr_type_node);
391       DECL_FIELD_CONTEXT (f_lsda) = sjlj_fc_type_node;
392
393 #ifdef DONT_USE_BUILTIN_SETJMP
394 #ifdef JMP_BUF_SIZE
395       tmp = build_int_cst (NULL_TREE, JMP_BUF_SIZE - 1, 0);
396 #else
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_cst (NULL_TREE,
402                            FIRST_PSEUDO_REGISTER + 2 - 1, 0);
403 #endif
404 #else
405       /* builtin_setjmp takes a pointer to 5 words.  */
406       tmp = build_int_cst (NULL_TREE,
407                            5 * BITS_PER_WORD / POINTER_SIZE - 1, 0);
408 #endif
409       tmp = build_index_type (tmp);
410       tmp = build_array_type (ptr_type_node, tmp);
411       f_jbuf = build_decl (FIELD_DECL, get_identifier ("__jbuf"), tmp);
412 #ifdef DONT_USE_BUILTIN_SETJMP
413       /* We don't know what the alignment requirements of the
414          runtime's jmp_buf has.  Overestimate.  */
415       DECL_ALIGN (f_jbuf) = BIGGEST_ALIGNMENT;
416       DECL_USER_ALIGN (f_jbuf) = 1;
417 #endif
418       DECL_FIELD_CONTEXT (f_jbuf) = sjlj_fc_type_node;
419
420       TYPE_FIELDS (sjlj_fc_type_node) = f_prev;
421       TREE_CHAIN (f_prev) = f_cs;
422       TREE_CHAIN (f_cs) = f_data;
423       TREE_CHAIN (f_data) = f_per;
424       TREE_CHAIN (f_per) = f_lsda;
425       TREE_CHAIN (f_lsda) = f_jbuf;
426
427       layout_type (sjlj_fc_type_node);
428
429       /* Cache the interesting field offsets so that we have
430          easy access from rtl.  */
431       sjlj_fc_call_site_ofs
432         = (tree_low_cst (DECL_FIELD_OFFSET (f_cs), 1)
433            + tree_low_cst (DECL_FIELD_BIT_OFFSET (f_cs), 1) / BITS_PER_UNIT);
434       sjlj_fc_data_ofs
435         = (tree_low_cst (DECL_FIELD_OFFSET (f_data), 1)
436            + tree_low_cst (DECL_FIELD_BIT_OFFSET (f_data), 1) / BITS_PER_UNIT);
437       sjlj_fc_personality_ofs
438         = (tree_low_cst (DECL_FIELD_OFFSET (f_per), 1)
439            + tree_low_cst (DECL_FIELD_BIT_OFFSET (f_per), 1) / BITS_PER_UNIT);
440       sjlj_fc_lsda_ofs
441         = (tree_low_cst (DECL_FIELD_OFFSET (f_lsda), 1)
442            + tree_low_cst (DECL_FIELD_BIT_OFFSET (f_lsda), 1) / BITS_PER_UNIT);
443       sjlj_fc_jbuf_ofs
444         = (tree_low_cst (DECL_FIELD_OFFSET (f_jbuf), 1)
445            + tree_low_cst (DECL_FIELD_BIT_OFFSET (f_jbuf), 1) / BITS_PER_UNIT);
446     }
447 }
448
449 void
450 init_eh_for_function (void)
451 {
452   cfun->eh = ggc_alloc_cleared (sizeof (struct eh_status));
453 }
454 \f
455 /* Routines to generate the exception tree somewhat directly.  
456    These are used from tree-eh.c when processing exception related
457    nodes during tree optimization.  */
458
459 static struct eh_region *
460 gen_eh_region (enum eh_region_type type, struct eh_region *outer)
461 {
462   struct eh_region *new;
463
464 #ifdef ENABLE_CHECKING
465   if (! doing_eh (0))
466     abort ();
467 #endif
468
469   /* Insert a new blank region as a leaf in the tree.  */
470   new = ggc_alloc_cleared (sizeof (*new));
471   new->type = type;
472   new->outer = outer;
473   if (outer)
474     {
475       new->next_peer = outer->inner;
476       outer->inner = new;
477     }
478   else
479     {
480       new->next_peer = cfun->eh->region_tree;
481       cfun->eh->region_tree = new;
482     }
483
484   new->region_number = ++cfun->eh->last_region_number;
485
486   return new;
487 }
488
489 struct eh_region *
490 gen_eh_region_cleanup (struct eh_region *outer, struct eh_region *prev_try)
491 {
492   struct eh_region *cleanup = gen_eh_region (ERT_CLEANUP, outer);
493   cleanup->u.cleanup.prev_try = prev_try;
494   return cleanup;
495 }
496
497 struct eh_region *
498 gen_eh_region_try (struct eh_region *outer)
499 {
500   return gen_eh_region (ERT_TRY, outer);
501 }
502
503 struct eh_region *
504 gen_eh_region_catch (struct eh_region *t, tree type_or_list)
505 {
506   struct eh_region *c, *l;
507   tree type_list, type_node;
508
509   /* Ensure to always end up with a type list to normalize further
510      processing, then register each type against the runtime types map.  */
511   type_list = type_or_list;
512   if (type_or_list)
513     {
514       if (TREE_CODE (type_or_list) != TREE_LIST)
515         type_list = tree_cons (NULL_TREE, type_or_list, NULL_TREE);
516
517       type_node = type_list;
518       for (; type_node; type_node = TREE_CHAIN (type_node))
519         add_type_for_runtime (TREE_VALUE (type_node));
520     }
521
522   c = gen_eh_region (ERT_CATCH, t->outer);
523   c->u.catch.type_list = type_list;
524   l = t->u.try.last_catch;
525   c->u.catch.prev_catch = l;
526   if (l)
527     l->u.catch.next_catch = c;
528   else
529     t->u.try.catch = c;
530   t->u.try.last_catch = c;
531
532   return c;
533 }
534
535 struct eh_region *
536 gen_eh_region_allowed (struct eh_region *outer, tree allowed)
537 {
538   struct eh_region *region = gen_eh_region (ERT_ALLOWED_EXCEPTIONS, outer);
539   region->u.allowed.type_list = allowed;
540
541   for (; allowed ; allowed = TREE_CHAIN (allowed))
542     add_type_for_runtime (TREE_VALUE (allowed));
543
544   return region;
545 }
546
547 struct eh_region *
548 gen_eh_region_must_not_throw (struct eh_region *outer)
549 {
550   return gen_eh_region (ERT_MUST_NOT_THROW, outer);
551 }
552
553 int
554 get_eh_region_number (struct eh_region *region)
555 {
556   return region->region_number;
557 }
558
559 bool
560 get_eh_region_may_contain_throw (struct eh_region *region)
561 {
562   return region->may_contain_throw;
563 }
564
565 tree
566 get_eh_region_tree_label (struct eh_region *region)
567 {
568   return region->tree_label;
569 }
570
571 void
572 set_eh_region_tree_label (struct eh_region *region, tree lab)
573 {
574   region->tree_label = lab;
575 }
576 \f
577 void
578 expand_resx_expr (tree exp)
579 {
580   int region_nr = TREE_INT_CST_LOW (TREE_OPERAND (exp, 0));
581   struct eh_region *reg = cfun->eh->region_array[region_nr];
582
583   reg->resume = emit_jump_insn (gen_rtx_RESX (VOIDmode, region_nr));
584   emit_barrier ();
585 }
586
587 /* Note that the current EH region (if any) may contain a throw, or a
588    call to a function which itself may contain a throw.  */
589
590 void
591 note_eh_region_may_contain_throw (struct eh_region *region)
592 {
593   while (region && !region->may_contain_throw)
594     {
595       region->may_contain_throw = 1;
596       region = region->outer;
597     }
598 }
599
600 void
601 note_current_region_may_contain_throw (void)
602 {
603   note_eh_region_may_contain_throw (cfun->eh->cur_region);
604 }
605
606
607 /* Return an rtl expression for a pointer to the exception object
608    within a handler.  */
609
610 rtx
611 get_exception_pointer (struct function *fun)
612 {
613   rtx exc_ptr = fun->eh->exc_ptr;
614   if (fun == cfun && ! exc_ptr)
615     {
616       exc_ptr = gen_reg_rtx (ptr_mode);
617       fun->eh->exc_ptr = exc_ptr;
618     }
619   return exc_ptr;
620 }
621
622 /* Return an rtl expression for the exception dispatch filter
623    within a handler.  */
624
625 rtx
626 get_exception_filter (struct function *fun)
627 {
628   rtx filter = fun->eh->filter;
629   if (fun == cfun && ! filter)
630     {
631       filter = gen_reg_rtx (targetm.eh_return_filter_mode ());
632       fun->eh->filter = filter;
633     }
634   return filter;
635 }
636 \f
637 /* This section is for the exception handling specific optimization pass.  */
638
639 /* Random access the exception region tree.  */
640
641 void
642 collect_eh_region_array (void)
643 {
644   struct eh_region **array, *i;
645
646   i = cfun->eh->region_tree;
647   if (! i)
648     return;
649
650   array = ggc_alloc_cleared ((cfun->eh->last_region_number + 1)
651                              * sizeof (*array));
652   cfun->eh->region_array = array;
653
654   while (1)
655     {
656       array[i->region_number] = i;
657
658       /* If there are sub-regions, process them.  */
659       if (i->inner)
660         i = i->inner;
661       /* If there are peers, process them.  */
662       else if (i->next_peer)
663         i = i->next_peer;
664       /* Otherwise, step back up the tree to the next peer.  */
665       else
666         {
667           do {
668             i = i->outer;
669             if (i == NULL)
670               return;
671           } while (i->next_peer == NULL);
672           i = i->next_peer;
673         }
674     }
675 }
676
677 static void
678 resolve_one_fixup_region (struct eh_region *fixup)
679 {
680   struct eh_region *cleanup, *real;
681   int j, n;
682
683   n = cfun->eh->last_region_number;
684   cleanup = 0;
685
686   for (j = 1; j <= n; ++j)
687     {
688       cleanup = cfun->eh->region_array[j];
689       if (cleanup && cleanup->type == ERT_CLEANUP
690           && cleanup->u.cleanup.exp == fixup->u.fixup.cleanup_exp)
691         break;
692     }
693   if (j > n)
694     abort ();
695
696   real = cleanup->outer;
697   if (real && real->type == ERT_FIXUP)
698     {
699       if (!real->u.fixup.resolved)
700         resolve_one_fixup_region (real);
701       real = real->u.fixup.real_region;
702     }
703
704   fixup->u.fixup.real_region = real;
705   fixup->u.fixup.resolved = true;
706 }
707
708 static void
709 resolve_fixup_regions (void)
710 {
711   int i, n = cfun->eh->last_region_number;
712
713   for (i = 1; i <= n; ++i)
714     {
715       struct eh_region *fixup = cfun->eh->region_array[i];
716
717       if (!fixup || fixup->type != ERT_FIXUP || fixup->u.fixup.resolved)
718         continue;
719
720       resolve_one_fixup_region (fixup);
721     }
722 }
723
724 /* Now that we've discovered what region actually encloses a fixup,
725    we can shuffle pointers and remove them from the tree.  */
726
727 static void
728 remove_fixup_regions (void)
729 {
730   int i;
731   rtx insn, note;
732   struct eh_region *fixup;
733
734   /* Walk the insn chain and adjust the REG_EH_REGION numbers
735      for instructions referencing fixup regions.  This is only
736      strictly necessary for fixup regions with no parent, but
737      doesn't hurt to do it for all regions.  */
738   for (insn = get_insns(); insn ; insn = NEXT_INSN (insn))
739     if (INSN_P (insn)
740         && (note = find_reg_note (insn, REG_EH_REGION, NULL))
741         && INTVAL (XEXP (note, 0)) > 0
742         && (fixup = cfun->eh->region_array[INTVAL (XEXP (note, 0))])
743         && fixup->type == ERT_FIXUP)
744       {
745         if (fixup->u.fixup.real_region)
746           XEXP (note, 0) = GEN_INT (fixup->u.fixup.real_region->region_number);
747         else
748           remove_note (insn, note);
749       }
750
751   /* Remove the fixup regions from the tree.  */
752   for (i = cfun->eh->last_region_number; i > 0; --i)
753     {
754       fixup = cfun->eh->region_array[i];
755       if (! fixup)
756         continue;
757
758       /* Allow GC to maybe free some memory.  */
759       if (fixup->type == ERT_CLEANUP)
760         fixup->u.cleanup.exp = NULL_TREE;
761
762       if (fixup->type != ERT_FIXUP)
763         continue;
764
765       if (fixup->inner)
766         {
767           struct eh_region *parent, *p, **pp;
768
769           parent = fixup->u.fixup.real_region;
770
771           /* Fix up the children's parent pointers; find the end of
772              the list.  */
773           for (p = fixup->inner; ; p = p->next_peer)
774             {
775               p->outer = parent;
776               if (! p->next_peer)
777                 break;
778             }
779
780           /* In the tree of cleanups, only outer-inner ordering matters.
781              So link the children back in anywhere at the correct level.  */
782           if (parent)
783             pp = &parent->inner;
784           else
785             pp = &cfun->eh->region_tree;
786           p->next_peer = *pp;
787           *pp = fixup->inner;
788           fixup->inner = NULL;
789         }
790
791       remove_eh_handler (fixup);
792     }
793 }
794
795 /* Remove all regions whose labels are not reachable from insns.  */
796
797 static void
798 remove_unreachable_regions (rtx insns)
799 {
800   int i, *uid_region_num;
801   bool *reachable;
802   struct eh_region *r;
803   rtx insn;
804
805   uid_region_num = xcalloc (get_max_uid (), sizeof(int));
806   reachable = xcalloc (cfun->eh->last_region_number + 1, sizeof(bool));
807
808   for (i = cfun->eh->last_region_number; i > 0; --i)
809     {
810       r = cfun->eh->region_array[i];
811       if (!r || r->region_number != i)
812         continue;
813
814       if (r->resume)
815         {
816           if (uid_region_num[INSN_UID (r->resume)])
817             abort ();
818           uid_region_num[INSN_UID (r->resume)] = i;
819         }
820       if (r->label)
821         {
822           if (uid_region_num[INSN_UID (r->label)])
823             abort ();
824           uid_region_num[INSN_UID (r->label)] = i;
825         }
826     }
827
828   for (insn = insns; insn; insn = NEXT_INSN (insn))
829     reachable[uid_region_num[INSN_UID (insn)]] = true;
830
831   for (i = cfun->eh->last_region_number; i > 0; --i)
832     {
833       r = cfun->eh->region_array[i];
834       if (r && r->region_number == i && !reachable[i])
835         {
836           bool kill_it = true;
837           switch (r->type)
838             {
839             case ERT_THROW:
840               /* Don't remove ERT_THROW regions if their outer region
841                  is reachable.  */
842               if (r->outer && reachable[r->outer->region_number])
843                 kill_it = false;
844               break;
845
846             case ERT_MUST_NOT_THROW:
847               /* MUST_NOT_THROW regions are implementable solely in the
848                  runtime, but their existence continues to affect calls
849                  within that region.  Never delete them here.  */
850               kill_it = false;
851               break;
852
853             case ERT_TRY:
854               {
855                 /* TRY regions are reachable if any of its CATCH regions
856                    are reachable.  */
857                 struct eh_region *c;
858                 for (c = r->u.try.catch; c ; c = c->u.catch.next_catch)
859                   if (reachable[c->region_number])
860                     {
861                       kill_it = false;
862                       break;
863                     }
864                 break;
865               }
866
867             default:
868               break;
869             }
870               
871           if (kill_it)
872             remove_eh_handler (r);
873         }
874     }
875
876   free (reachable);
877   free (uid_region_num);
878 }
879
880 /* Turn NOTE_INSN_EH_REGION notes into REG_EH_REGION notes for each
881    can_throw instruction in the region.  */
882
883 static void
884 convert_from_eh_region_ranges_1 (rtx *pinsns, int *orig_sp, int cur)
885 {
886   int *sp = orig_sp;
887   rtx insn, next;
888
889   for (insn = *pinsns; insn ; insn = next)
890     {
891       next = NEXT_INSN (insn);
892       if (NOTE_P (insn))
893         {
894           int kind = NOTE_LINE_NUMBER (insn);
895           if (kind == NOTE_INSN_EH_REGION_BEG
896               || kind == NOTE_INSN_EH_REGION_END)
897             {
898               if (kind == NOTE_INSN_EH_REGION_BEG)
899                 {
900                   struct eh_region *r;
901
902                   *sp++ = cur;
903                   cur = NOTE_EH_HANDLER (insn);
904
905                   r = cfun->eh->region_array[cur];
906                   if (r->type == ERT_FIXUP)
907                     {
908                       r = r->u.fixup.real_region;
909                       cur = r ? r->region_number : 0;
910                     }
911                   else if (r->type == ERT_CATCH)
912                     {
913                       r = r->outer;
914                       cur = r ? r->region_number : 0;
915                     }
916                 }
917               else
918                 cur = *--sp;
919
920               if (insn == *pinsns)
921                 *pinsns = next;
922               remove_insn (insn);
923               continue;
924             }
925         }
926       else if (INSN_P (insn))
927         {
928           if (cur > 0
929               && ! find_reg_note (insn, REG_EH_REGION, NULL_RTX)
930               /* Calls can always potentially throw exceptions, unless
931                  they have a REG_EH_REGION note with a value of 0 or less.
932                  Which should be the only possible kind so far.  */
933               && (CALL_P (insn)
934                   /* If we wanted exceptions for non-call insns, then
935                      any may_trap_p instruction could throw.  */
936                   || (flag_non_call_exceptions
937                       && GET_CODE (PATTERN (insn)) != CLOBBER
938                       && GET_CODE (PATTERN (insn)) != USE
939                       && may_trap_p (PATTERN (insn)))))
940             {
941               REG_NOTES (insn) = alloc_EXPR_LIST (REG_EH_REGION, GEN_INT (cur),
942                                                   REG_NOTES (insn));
943             }
944         }
945     }
946
947   if (sp != orig_sp)
948     abort ();
949 }
950
951 static void
952 collect_rtl_labels_from_trees (void)
953 {
954   int i, n = cfun->eh->last_region_number;
955   for (i = 1; i <= n; ++i)
956     {
957       struct eh_region *reg = cfun->eh->region_array[i];
958       if (reg && reg->tree_label)
959         reg->label = DECL_RTL_IF_SET (reg->tree_label);
960     }
961 }
962
963 void
964 convert_from_eh_region_ranges (void)
965 {
966   rtx insns = get_insns ();
967
968   if (cfun->eh->region_array)
969     {
970       /* If the region array already exists, assume we're coming from
971          optimize_function_tree.  In this case all we need to do is
972          collect the rtl labels that correspond to the tree labels
973          that we allocated earlier.  */
974       collect_rtl_labels_from_trees ();
975     }
976   else
977     {
978       int *stack;
979
980       collect_eh_region_array ();
981       resolve_fixup_regions ();
982
983       stack = xmalloc (sizeof (int) * (cfun->eh->last_region_number + 1));
984       convert_from_eh_region_ranges_1 (&insns, stack, 0);
985       free (stack);
986
987       remove_fixup_regions ();
988     }
989
990   remove_unreachable_regions (insns);
991 }
992
993 static void
994 add_ehl_entry (rtx label, struct eh_region *region)
995 {
996   struct ehl_map_entry **slot, *entry;
997
998   LABEL_PRESERVE_P (label) = 1;
999
1000   entry = ggc_alloc (sizeof (*entry));
1001   entry->label = label;
1002   entry->region = region;
1003
1004   slot = (struct ehl_map_entry **)
1005     htab_find_slot (cfun->eh->exception_handler_label_map, entry, INSERT);
1006
1007   /* Before landing pad creation, each exception handler has its own
1008      label.  After landing pad creation, the exception handlers may
1009      share landing pads.  This is ok, since maybe_remove_eh_handler
1010      only requires the 1-1 mapping before landing pad creation.  */
1011   if (*slot && !cfun->eh->built_landing_pads)
1012     abort ();
1013
1014   *slot = entry;
1015 }
1016
1017 void
1018 find_exception_handler_labels (void)
1019 {
1020   int i;
1021
1022   if (cfun->eh->exception_handler_label_map)
1023     htab_empty (cfun->eh->exception_handler_label_map);
1024   else
1025     {
1026       /* ??? The expansion factor here (3/2) must be greater than the htab
1027          occupancy factor (4/3) to avoid unnecessary resizing.  */
1028       cfun->eh->exception_handler_label_map
1029         = htab_create_ggc (cfun->eh->last_region_number * 3 / 2,
1030                            ehl_hash, ehl_eq, NULL);
1031     }
1032
1033   if (cfun->eh->region_tree == NULL)
1034     return;
1035
1036   for (i = cfun->eh->last_region_number; i > 0; --i)
1037     {
1038       struct eh_region *region = cfun->eh->region_array[i];
1039       rtx lab;
1040
1041       if (! region || region->region_number != i)
1042         continue;
1043       if (cfun->eh->built_landing_pads)
1044         lab = region->landing_pad;
1045       else
1046         lab = region->label;
1047
1048       if (lab)
1049         add_ehl_entry (lab, region);
1050     }
1051
1052   /* For sjlj exceptions, need the return label to remain live until
1053      after landing pad generation.  */
1054   if (USING_SJLJ_EXCEPTIONS && ! cfun->eh->built_landing_pads)
1055     add_ehl_entry (return_label, NULL);
1056 }
1057
1058 bool
1059 current_function_has_exception_handlers (void)
1060 {
1061   int i;
1062
1063   for (i = cfun->eh->last_region_number; i > 0; --i)
1064     {
1065       struct eh_region *region = cfun->eh->region_array[i];
1066
1067       if (! region || region->region_number != i)
1068         continue;
1069       if (region->type != ERT_THROW)
1070         return true;
1071     }
1072
1073   return false;
1074 }
1075 \f
1076 static struct eh_region *
1077 duplicate_eh_region_1 (struct eh_region *o, struct inline_remap *map)
1078 {
1079   struct eh_region *n = ggc_alloc_cleared (sizeof (struct eh_region));
1080
1081   n->region_number = o->region_number + cfun->eh->last_region_number;
1082   n->type = o->type;
1083
1084   switch (n->type)
1085     {
1086     case ERT_CLEANUP:
1087     case ERT_MUST_NOT_THROW:
1088       break;
1089
1090     case ERT_TRY:
1091       if (o->u.try.continue_label)
1092         n->u.try.continue_label
1093           = get_label_from_map (map,
1094                                 CODE_LABEL_NUMBER (o->u.try.continue_label));
1095       break;
1096
1097     case ERT_CATCH:
1098       n->u.catch.type_list = o->u.catch.type_list;
1099       break;
1100
1101     case ERT_ALLOWED_EXCEPTIONS:
1102       n->u.allowed.type_list = o->u.allowed.type_list;
1103       break;
1104
1105     case ERT_THROW:
1106       n->u.throw.type = o->u.throw.type;
1107
1108     default:
1109       abort ();
1110     }
1111
1112   if (o->label)
1113     n->label = get_label_from_map (map, CODE_LABEL_NUMBER (o->label));
1114   if (o->resume)
1115     {
1116       n->resume = map->insn_map[INSN_UID (o->resume)];
1117       if (n->resume == NULL)
1118         abort ();
1119     }
1120
1121   return n;
1122 }
1123
1124 static void
1125 duplicate_eh_region_2 (struct eh_region *o, struct eh_region **n_array)
1126 {
1127   struct eh_region *n = n_array[o->region_number];
1128
1129   switch (n->type)
1130     {
1131     case ERT_TRY:
1132       n->u.try.catch = n_array[o->u.try.catch->region_number];
1133       n->u.try.last_catch = n_array[o->u.try.last_catch->region_number];
1134       break;
1135
1136     case ERT_CATCH:
1137       if (o->u.catch.next_catch)
1138         n->u.catch.next_catch = n_array[o->u.catch.next_catch->region_number];
1139       if (o->u.catch.prev_catch)
1140         n->u.catch.prev_catch = n_array[o->u.catch.prev_catch->region_number];
1141       break;
1142
1143     default:
1144       break;
1145     }
1146
1147   if (o->outer)
1148     n->outer = n_array[o->outer->region_number];
1149   if (o->inner)
1150     n->inner = n_array[o->inner->region_number];
1151   if (o->next_peer)
1152     n->next_peer = n_array[o->next_peer->region_number];
1153 }
1154
1155 int
1156 duplicate_eh_regions (struct function *ifun, struct inline_remap *map)
1157 {
1158   int ifun_last_region_number = ifun->eh->last_region_number;
1159   struct eh_region **n_array, *root, *cur;
1160   int i;
1161
1162   if (ifun_last_region_number == 0)
1163     return 0;
1164
1165   n_array = xcalloc (ifun_last_region_number + 1, sizeof (*n_array));
1166
1167   for (i = 1; i <= ifun_last_region_number; ++i)
1168     {
1169       cur = ifun->eh->region_array[i];
1170       if (!cur || cur->region_number != i)
1171         continue;
1172       n_array[i] = duplicate_eh_region_1 (cur, map);
1173     }
1174   for (i = 1; i <= ifun_last_region_number; ++i)
1175     {
1176       cur = ifun->eh->region_array[i];
1177       if (!cur || cur->region_number != i)
1178         continue;
1179       duplicate_eh_region_2 (cur, n_array);
1180     }
1181
1182   root = n_array[ifun->eh->region_tree->region_number];
1183   cur = cfun->eh->cur_region;
1184   if (cur)
1185     {
1186       struct eh_region *p = cur->inner;
1187       if (p)
1188         {
1189           while (p->next_peer)
1190             p = p->next_peer;
1191           p->next_peer = root;
1192         }
1193       else
1194         cur->inner = root;
1195
1196       for (i = 1; i <= ifun_last_region_number; ++i)
1197         if (n_array[i] && n_array[i]->outer == NULL)
1198           n_array[i]->outer = cur;
1199     }
1200   else
1201     {
1202       struct eh_region *p = cfun->eh->region_tree;
1203       if (p)
1204         {
1205           while (p->next_peer)
1206             p = p->next_peer;
1207           p->next_peer = root;
1208         }
1209       else
1210         cfun->eh->region_tree = root;
1211     }
1212
1213   free (n_array);
1214
1215   i = cfun->eh->last_region_number;
1216   cfun->eh->last_region_number = i + ifun_last_region_number;
1217   return i;
1218 }
1219
1220 \f
1221 static int
1222 t2r_eq (const void *pentry, const void *pdata)
1223 {
1224   tree entry = (tree) pentry;
1225   tree data = (tree) pdata;
1226
1227   return TREE_PURPOSE (entry) == data;
1228 }
1229
1230 static hashval_t
1231 t2r_hash (const void *pentry)
1232 {
1233   tree entry = (tree) pentry;
1234   return TREE_HASH (TREE_PURPOSE (entry));
1235 }
1236
1237 static void
1238 add_type_for_runtime (tree type)
1239 {
1240   tree *slot;
1241
1242   slot = (tree *) htab_find_slot_with_hash (type_to_runtime_map, type,
1243                                             TREE_HASH (type), INSERT);
1244   if (*slot == NULL)
1245     {
1246       tree runtime = (*lang_eh_runtime_type) (type);
1247       *slot = tree_cons (type, runtime, NULL_TREE);
1248     }
1249 }
1250
1251 static tree
1252 lookup_type_for_runtime (tree type)
1253 {
1254   tree *slot;
1255
1256   slot = (tree *) htab_find_slot_with_hash (type_to_runtime_map, type,
1257                                             TREE_HASH (type), NO_INSERT);
1258
1259   /* We should have always inserted the data earlier.  */
1260   return TREE_VALUE (*slot);
1261 }
1262
1263 \f
1264 /* Represent an entry in @TTypes for either catch actions
1265    or exception filter actions.  */
1266 struct ttypes_filter GTY(())
1267 {
1268   tree t;
1269   int filter;
1270 };
1271
1272 /* Compare ENTRY (a ttypes_filter entry in the hash table) with DATA
1273    (a tree) for a @TTypes type node we are thinking about adding.  */
1274
1275 static int
1276 ttypes_filter_eq (const void *pentry, const void *pdata)
1277 {
1278   const struct ttypes_filter *entry = (const struct ttypes_filter *) pentry;
1279   tree data = (tree) pdata;
1280
1281   return entry->t == data;
1282 }
1283
1284 static hashval_t
1285 ttypes_filter_hash (const void *pentry)
1286 {
1287   const struct ttypes_filter *entry = (const struct ttypes_filter *) pentry;
1288   return TREE_HASH (entry->t);
1289 }
1290
1291 /* Compare ENTRY with DATA (both struct ttypes_filter) for a @TTypes
1292    exception specification list we are thinking about adding.  */
1293 /* ??? Currently we use the type lists in the order given.  Someone
1294    should put these in some canonical order.  */
1295
1296 static int
1297 ehspec_filter_eq (const void *pentry, const void *pdata)
1298 {
1299   const struct ttypes_filter *entry = (const struct ttypes_filter *) pentry;
1300   const struct ttypes_filter *data = (const struct ttypes_filter *) pdata;
1301
1302   return type_list_equal (entry->t, data->t);
1303 }
1304
1305 /* Hash function for exception specification lists.  */
1306
1307 static hashval_t
1308 ehspec_filter_hash (const void *pentry)
1309 {
1310   const struct ttypes_filter *entry = (const struct ttypes_filter *) pentry;
1311   hashval_t h = 0;
1312   tree list;
1313
1314   for (list = entry->t; list ; list = TREE_CHAIN (list))
1315     h = (h << 5) + (h >> 27) + TREE_HASH (TREE_VALUE (list));
1316   return h;
1317 }
1318
1319 /* Add TYPE (which may be NULL) to cfun->eh->ttype_data, using TYPES_HASH
1320    to speed up the search.  Return the filter value to be used.  */
1321
1322 static int
1323 add_ttypes_entry (htab_t ttypes_hash, tree type)
1324 {
1325   struct ttypes_filter **slot, *n;
1326
1327   slot = (struct ttypes_filter **)
1328     htab_find_slot_with_hash (ttypes_hash, type, TREE_HASH (type), INSERT);
1329
1330   if ((n = *slot) == NULL)
1331     {
1332       /* Filter value is a 1 based table index.  */
1333
1334       n = xmalloc (sizeof (*n));
1335       n->t = type;
1336       n->filter = VARRAY_ACTIVE_SIZE (cfun->eh->ttype_data) + 1;
1337       *slot = n;
1338
1339       VARRAY_PUSH_TREE (cfun->eh->ttype_data, type);
1340     }
1341
1342   return n->filter;
1343 }
1344
1345 /* Add LIST to cfun->eh->ehspec_data, using EHSPEC_HASH and TYPES_HASH
1346    to speed up the search.  Return the filter value to be used.  */
1347
1348 static int
1349 add_ehspec_entry (htab_t ehspec_hash, htab_t ttypes_hash, tree list)
1350 {
1351   struct ttypes_filter **slot, *n;
1352   struct ttypes_filter dummy;
1353
1354   dummy.t = list;
1355   slot = (struct ttypes_filter **)
1356     htab_find_slot (ehspec_hash, &dummy, INSERT);
1357
1358   if ((n = *slot) == NULL)
1359     {
1360       /* Filter value is a -1 based byte index into a uleb128 buffer.  */
1361
1362       n = xmalloc (sizeof (*n));
1363       n->t = list;
1364       n->filter = -(VARRAY_ACTIVE_SIZE (cfun->eh->ehspec_data) + 1);
1365       *slot = n;
1366
1367       /* Look up each type in the list and encode its filter
1368          value as a uleb128.  Terminate the list with 0.  */
1369       for (; list ; list = TREE_CHAIN (list))
1370         push_uleb128 (&cfun->eh->ehspec_data,
1371                       add_ttypes_entry (ttypes_hash, TREE_VALUE (list)));
1372       VARRAY_PUSH_UCHAR (cfun->eh->ehspec_data, 0);
1373     }
1374
1375   return n->filter;
1376 }
1377
1378 /* Generate the action filter values to be used for CATCH and
1379    ALLOWED_EXCEPTIONS regions.  When using dwarf2 exception regions,
1380    we use lots of landing pads, and so every type or list can share
1381    the same filter value, which saves table space.  */
1382
1383 static void
1384 assign_filter_values (void)
1385 {
1386   int i;
1387   htab_t ttypes, ehspec;
1388
1389   VARRAY_TREE_INIT (cfun->eh->ttype_data, 16, "ttype_data");
1390   VARRAY_UCHAR_INIT (cfun->eh->ehspec_data, 64, "ehspec_data");
1391
1392   ttypes = htab_create (31, ttypes_filter_hash, ttypes_filter_eq, free);
1393   ehspec = htab_create (31, ehspec_filter_hash, ehspec_filter_eq, free);
1394
1395   for (i = cfun->eh->last_region_number; i > 0; --i)
1396     {
1397       struct eh_region *r = cfun->eh->region_array[i];
1398
1399       /* Mind we don't process a region more than once.  */
1400       if (!r || r->region_number != i)
1401         continue;
1402
1403       switch (r->type)
1404         {
1405         case ERT_CATCH:
1406           /* Whatever type_list is (NULL or true list), we build a list
1407              of filters for the region.  */
1408           r->u.catch.filter_list = NULL_TREE;
1409
1410           if (r->u.catch.type_list != NULL)
1411             {
1412               /* Get a filter value for each of the types caught and store
1413                  them in the region's dedicated list.  */
1414               tree tp_node = r->u.catch.type_list;
1415
1416               for (;tp_node; tp_node = TREE_CHAIN (tp_node))
1417                 {
1418                   int flt = add_ttypes_entry (ttypes, TREE_VALUE (tp_node));
1419                   tree flt_node = build_int_cst (NULL_TREE, flt, 0);
1420
1421                   r->u.catch.filter_list
1422                     = tree_cons (NULL_TREE, flt_node, r->u.catch.filter_list);
1423                 }
1424             }
1425           else
1426             {
1427               /* Get a filter value for the NULL list also since it will need
1428                  an action record anyway.  */
1429               int flt = add_ttypes_entry (ttypes, NULL);
1430               tree flt_node = build_int_cst (NULL_TREE, flt, 0);
1431
1432               r->u.catch.filter_list
1433                 = tree_cons (NULL_TREE, flt_node, r->u.catch.filter_list);
1434             }
1435
1436           break;
1437
1438         case ERT_ALLOWED_EXCEPTIONS:
1439           r->u.allowed.filter
1440             = add_ehspec_entry (ehspec, ttypes, r->u.allowed.type_list);
1441           break;
1442
1443         default:
1444           break;
1445         }
1446     }
1447
1448   htab_delete (ttypes);
1449   htab_delete (ehspec);
1450 }
1451
1452 /* Emit SEQ into basic block just before INSN (that is assumed to be
1453    first instruction of some existing BB and return the newly
1454    produced block.  */
1455 static basic_block
1456 emit_to_new_bb_before (rtx seq, rtx insn)
1457 {
1458   rtx last;
1459   basic_block bb;
1460   edge e;
1461
1462   /* If there happens to be an fallthru edge (possibly created by cleanup_cfg
1463      call), we don't want it to go into newly created landing pad or other EH 
1464      construct.  */
1465   for (e = BLOCK_FOR_INSN (insn)->pred; e; e = e->pred_next)
1466     if (e->flags & EDGE_FALLTHRU)
1467       force_nonfallthru (e);
1468   last = emit_insn_before (seq, insn);
1469   if (BARRIER_P (last))
1470     last = PREV_INSN (last);
1471   bb = create_basic_block (seq, last, BLOCK_FOR_INSN (insn)->prev_bb);
1472   update_bb_for_insn (bb);
1473   bb->flags |= BB_SUPERBLOCK;
1474   return bb;
1475 }
1476
1477 /* Generate the code to actually handle exceptions, which will follow the
1478    landing pads.  */
1479
1480 static void
1481 build_post_landing_pads (void)
1482 {
1483   int i;
1484
1485   for (i = cfun->eh->last_region_number; i > 0; --i)
1486     {
1487       struct eh_region *region = cfun->eh->region_array[i];
1488       rtx seq;
1489
1490       /* Mind we don't process a region more than once.  */
1491       if (!region || region->region_number != i)
1492         continue;
1493
1494       switch (region->type)
1495         {
1496         case ERT_TRY:
1497           /* ??? Collect the set of all non-overlapping catch handlers
1498                all the way up the chain until blocked by a cleanup.  */
1499           /* ??? Outer try regions can share landing pads with inner
1500              try regions if the types are completely non-overlapping,
1501              and there are no intervening cleanups.  */
1502
1503           region->post_landing_pad = gen_label_rtx ();
1504
1505           start_sequence ();
1506
1507           emit_label (region->post_landing_pad);
1508
1509           /* ??? It is mighty inconvenient to call back into the
1510              switch statement generation code in expand_end_case.
1511              Rapid prototyping sez a sequence of ifs.  */
1512           {
1513             struct eh_region *c;
1514             for (c = region->u.try.catch; c ; c = c->u.catch.next_catch)
1515               {
1516                 if (c->u.catch.type_list == NULL)
1517                   emit_jump (c->label);
1518                 else
1519                   {
1520                     /* Need for one cmp/jump per type caught. Each type
1521                        list entry has a matching entry in the filter list
1522                        (see assign_filter_values).  */
1523                     tree tp_node = c->u.catch.type_list;
1524                     tree flt_node = c->u.catch.filter_list;
1525
1526                     for (; tp_node; )
1527                       {
1528                         emit_cmp_and_jump_insns
1529                           (cfun->eh->filter,
1530                            GEN_INT (tree_low_cst (TREE_VALUE (flt_node), 0)),
1531                            EQ, NULL_RTX, 
1532                            targetm.eh_return_filter_mode (), 0, c->label);
1533
1534                         tp_node = TREE_CHAIN (tp_node);
1535                         flt_node = TREE_CHAIN (flt_node);
1536                       }
1537                   }
1538               }
1539           }
1540
1541           /* We delay the generation of the _Unwind_Resume until we generate
1542              landing pads.  We emit a marker here so as to get good control
1543              flow data in the meantime.  */
1544           region->resume
1545             = emit_jump_insn (gen_rtx_RESX (VOIDmode, region->region_number));
1546           emit_barrier ();
1547
1548           seq = get_insns ();
1549           end_sequence ();
1550
1551           emit_to_new_bb_before (seq, region->u.try.catch->label);
1552
1553           break;
1554
1555         case ERT_ALLOWED_EXCEPTIONS:
1556           region->post_landing_pad = gen_label_rtx ();
1557
1558           start_sequence ();
1559
1560           emit_label (region->post_landing_pad);
1561
1562           emit_cmp_and_jump_insns (cfun->eh->filter,
1563                                    GEN_INT (region->u.allowed.filter),
1564                                    EQ, NULL_RTX, 
1565                                    targetm.eh_return_filter_mode (), 0, region->label);
1566
1567           /* We delay the generation of the _Unwind_Resume until we generate
1568              landing pads.  We emit a marker here so as to get good control
1569              flow data in the meantime.  */
1570           region->resume
1571             = emit_jump_insn (gen_rtx_RESX (VOIDmode, region->region_number));
1572           emit_barrier ();
1573
1574           seq = get_insns ();
1575           end_sequence ();
1576
1577           emit_to_new_bb_before (seq, region->label);
1578           break;
1579
1580         case ERT_CLEANUP:
1581         case ERT_MUST_NOT_THROW:
1582           region->post_landing_pad = region->label;
1583           break;
1584
1585         case ERT_CATCH:
1586         case ERT_THROW:
1587           /* Nothing to do.  */
1588           break;
1589
1590         default:
1591           abort ();
1592         }
1593     }
1594 }
1595
1596 /* Replace RESX patterns with jumps to the next handler if any, or calls to
1597    _Unwind_Resume otherwise.  */
1598
1599 static void
1600 connect_post_landing_pads (void)
1601 {
1602   int i;
1603
1604   for (i = cfun->eh->last_region_number; i > 0; --i)
1605     {
1606       struct eh_region *region = cfun->eh->region_array[i];
1607       struct eh_region *outer;
1608       rtx seq;
1609       rtx barrier;
1610
1611       /* Mind we don't process a region more than once.  */
1612       if (!region || region->region_number != i)
1613         continue;
1614
1615       /* If there is no RESX, or it has been deleted by flow, there's
1616          nothing to fix up.  */
1617       if (! region->resume || INSN_DELETED_P (region->resume))
1618         continue;
1619
1620       /* Search for another landing pad in this function.  */
1621       for (outer = region->outer; outer ; outer = outer->outer)
1622         if (outer->post_landing_pad)
1623           break;
1624
1625       start_sequence ();
1626
1627       if (outer)
1628         {
1629           edge e;
1630           basic_block src, dest;
1631
1632           emit_jump (outer->post_landing_pad);
1633           src = BLOCK_FOR_INSN (region->resume);
1634           dest = BLOCK_FOR_INSN (outer->post_landing_pad);
1635           while (src->succ)
1636             remove_edge (src->succ);
1637           e = make_edge (src, dest, 0);
1638           e->probability = REG_BR_PROB_BASE;
1639           e->count = src->count;
1640         }
1641       else
1642         {
1643           emit_library_call (unwind_resume_libfunc, LCT_THROW,
1644                              VOIDmode, 1, cfun->eh->exc_ptr, ptr_mode);
1645
1646           /* What we just emitted was a throwing libcall, so it got a
1647              barrier automatically added after it.  If the last insn in
1648              the libcall sequence isn't the barrier, it's because the
1649              target emits multiple insns for a call, and there are insns
1650              after the actual call insn (which are redundant and would be
1651              optimized away).  The barrier is inserted exactly after the
1652              call insn, so let's go get that and delete the insns after
1653              it, because below we need the barrier to be the last insn in
1654              the sequence.  */
1655           delete_insns_since (NEXT_INSN (last_call_insn ()));
1656         }
1657
1658       seq = get_insns ();
1659       end_sequence ();
1660       barrier = emit_insn_before (seq, region->resume);
1661       /* Avoid duplicate barrier.  */
1662       if (!BARRIER_P (barrier))
1663         abort ();
1664       delete_insn (barrier);
1665       delete_insn (region->resume);
1666
1667       /* ??? From tree-ssa we can wind up with catch regions whose
1668          label is not instantiated, but whose resx is present.  Now
1669          that we've dealt with the resx, kill the region.  */
1670       if (region->label == NULL && region->type == ERT_CLEANUP)
1671         remove_eh_handler (region);
1672     }
1673 }
1674
1675 \f
1676 static void
1677 dw2_build_landing_pads (void)
1678 {
1679   int i;
1680   unsigned int j;
1681
1682   for (i = cfun->eh->last_region_number; i > 0; --i)
1683     {
1684       struct eh_region *region = cfun->eh->region_array[i];
1685       rtx seq;
1686       basic_block bb;
1687       bool clobbers_hard_regs = false;
1688       edge e;
1689
1690       /* Mind we don't process a region more than once.  */
1691       if (!region || region->region_number != i)
1692         continue;
1693
1694       if (region->type != ERT_CLEANUP
1695           && region->type != ERT_TRY
1696           && region->type != ERT_ALLOWED_EXCEPTIONS)
1697         continue;
1698
1699       start_sequence ();
1700
1701       region->landing_pad = gen_label_rtx ();
1702       emit_label (region->landing_pad);
1703
1704 #ifdef HAVE_exception_receiver
1705       if (HAVE_exception_receiver)
1706         emit_insn (gen_exception_receiver ());
1707       else
1708 #endif
1709 #ifdef HAVE_nonlocal_goto_receiver
1710         if (HAVE_nonlocal_goto_receiver)
1711           emit_insn (gen_nonlocal_goto_receiver ());
1712         else
1713 #endif
1714           { /* Nothing */ }
1715
1716       /* If the eh_return data registers are call-saved, then we
1717          won't have considered them clobbered from the call that
1718          threw.  Kill them now.  */
1719       for (j = 0; ; ++j)
1720         {
1721           unsigned r = EH_RETURN_DATA_REGNO (j);
1722           if (r == INVALID_REGNUM)
1723             break;
1724           if (! call_used_regs[r])
1725             {
1726               emit_insn (gen_rtx_CLOBBER (VOIDmode, gen_rtx_REG (Pmode, r)));
1727               clobbers_hard_regs = true;
1728             }
1729         }
1730
1731       if (clobbers_hard_regs)
1732         {
1733           /* @@@ This is a kludge.  Not all machine descriptions define a
1734              blockage insn, but we must not allow the code we just generated
1735              to be reordered by scheduling.  So emit an ASM_INPUT to act as
1736              blockage insn.  */
1737           emit_insn (gen_rtx_ASM_INPUT (VOIDmode, ""));
1738         }
1739
1740       emit_move_insn (cfun->eh->exc_ptr,
1741                       gen_rtx_REG (ptr_mode, EH_RETURN_DATA_REGNO (0)));
1742       emit_move_insn (cfun->eh->filter,
1743                       gen_rtx_REG (targetm.eh_return_filter_mode (), 
1744                                    EH_RETURN_DATA_REGNO (1)));
1745
1746       seq = get_insns ();
1747       end_sequence ();
1748
1749       bb = emit_to_new_bb_before (seq, region->post_landing_pad);
1750       e = make_edge (bb, bb->next_bb, EDGE_FALLTHRU);
1751       e->count = bb->count;
1752       e->probability = REG_BR_PROB_BASE;
1753     }
1754 }
1755
1756 \f
1757 struct sjlj_lp_info
1758 {
1759   int directly_reachable;
1760   int action_index;
1761   int dispatch_index;
1762   int call_site_index;
1763 };
1764
1765 static bool
1766 sjlj_find_directly_reachable_regions (struct sjlj_lp_info *lp_info)
1767 {
1768   rtx insn;
1769   bool found_one = false;
1770
1771   for (insn = get_insns (); insn ; insn = NEXT_INSN (insn))
1772     {
1773       struct eh_region *region;
1774       enum reachable_code rc;
1775       tree type_thrown;
1776       rtx note;
1777
1778       if (! INSN_P (insn))
1779         continue;
1780
1781       note = find_reg_note (insn, REG_EH_REGION, NULL_RTX);
1782       if (!note || INTVAL (XEXP (note, 0)) <= 0)
1783         continue;
1784
1785       region = cfun->eh->region_array[INTVAL (XEXP (note, 0))];
1786
1787       type_thrown = NULL_TREE;
1788       if (region->type == ERT_THROW)
1789         {
1790           type_thrown = region->u.throw.type;
1791           region = region->outer;
1792         }
1793
1794       /* Find the first containing region that might handle the exception.
1795          That's the landing pad to which we will transfer control.  */
1796       rc = RNL_NOT_CAUGHT;
1797       for (; region; region = region->outer)
1798         {
1799           rc = reachable_next_level (region, type_thrown, NULL);
1800           if (rc != RNL_NOT_CAUGHT)
1801             break;
1802         }
1803       if (rc == RNL_MAYBE_CAUGHT || rc == RNL_CAUGHT)
1804         {
1805           lp_info[region->region_number].directly_reachable = 1;
1806           found_one = true;
1807         }
1808     }
1809
1810   return found_one;
1811 }
1812
1813 static void
1814 sjlj_assign_call_site_values (rtx dispatch_label, struct sjlj_lp_info *lp_info)
1815 {
1816   htab_t ar_hash;
1817   int i, index;
1818
1819   /* First task: build the action table.  */
1820
1821   VARRAY_UCHAR_INIT (cfun->eh->action_record_data, 64, "action_record_data");
1822   ar_hash = htab_create (31, action_record_hash, action_record_eq, free);
1823
1824   for (i = cfun->eh->last_region_number; i > 0; --i)
1825     if (lp_info[i].directly_reachable)
1826       {
1827         struct eh_region *r = cfun->eh->region_array[i];
1828         r->landing_pad = dispatch_label;
1829         lp_info[i].action_index = collect_one_action_chain (ar_hash, r);
1830         if (lp_info[i].action_index != -1)
1831           cfun->uses_eh_lsda = 1;
1832       }
1833
1834   htab_delete (ar_hash);
1835
1836   /* Next: assign dispatch values.  In dwarf2 terms, this would be the
1837      landing pad label for the region.  For sjlj though, there is one
1838      common landing pad from which we dispatch to the post-landing pads.
1839
1840      A region receives a dispatch index if it is directly reachable
1841      and requires in-function processing.  Regions that share post-landing
1842      pads may share dispatch indices.  */
1843   /* ??? Post-landing pad sharing doesn't actually happen at the moment
1844      (see build_post_landing_pads) so we don't bother checking for it.  */
1845
1846   index = 0;
1847   for (i = cfun->eh->last_region_number; i > 0; --i)
1848     if (lp_info[i].directly_reachable)
1849       lp_info[i].dispatch_index = index++;
1850
1851   /* Finally: assign call-site values.  If dwarf2 terms, this would be
1852      the region number assigned by convert_to_eh_region_ranges, but
1853      handles no-action and must-not-throw differently.  */
1854
1855   call_site_base = 1;
1856   for (i = cfun->eh->last_region_number; i > 0; --i)
1857     if (lp_info[i].directly_reachable)
1858       {
1859         int action = lp_info[i].action_index;
1860
1861         /* Map must-not-throw to otherwise unused call-site index 0.  */
1862         if (action == -2)
1863           index = 0;
1864         /* Map no-action to otherwise unused call-site index -1.  */
1865         else if (action == -1)
1866           index = -1;
1867         /* Otherwise, look it up in the table.  */
1868         else
1869           index = add_call_site (GEN_INT (lp_info[i].dispatch_index), action);
1870
1871         lp_info[i].call_site_index = index;
1872       }
1873 }
1874
1875 static void
1876 sjlj_mark_call_sites (struct sjlj_lp_info *lp_info)
1877 {
1878   int last_call_site = -2;
1879   rtx insn, mem;
1880
1881   for (insn = get_insns (); insn ; insn = NEXT_INSN (insn))
1882     {
1883       struct eh_region *region;
1884       int this_call_site;
1885       rtx note, before, p;
1886
1887       /* Reset value tracking at extended basic block boundaries.  */
1888       if (LABEL_P (insn))
1889         last_call_site = -2;
1890
1891       if (! INSN_P (insn))
1892         continue;
1893
1894       note = find_reg_note (insn, REG_EH_REGION, NULL_RTX);
1895       if (!note)
1896         {
1897           /* Calls (and trapping insns) without notes are outside any
1898              exception handling region in this function.  Mark them as
1899              no action.  */
1900           if (CALL_P (insn)
1901               || (flag_non_call_exceptions
1902                   && may_trap_p (PATTERN (insn))))
1903             this_call_site = -1;
1904           else
1905             continue;
1906         }
1907       else
1908         {
1909           /* Calls that are known to not throw need not be marked.  */
1910           if (INTVAL (XEXP (note, 0)) <= 0)
1911             continue;
1912
1913           region = cfun->eh->region_array[INTVAL (XEXP (note, 0))];
1914           this_call_site = lp_info[region->region_number].call_site_index;
1915         }
1916
1917       if (this_call_site == last_call_site)
1918         continue;
1919
1920       /* Don't separate a call from it's argument loads.  */
1921       before = insn;
1922       if (CALL_P (insn))
1923         before = find_first_parameter_load (insn, NULL_RTX);
1924
1925       start_sequence ();
1926       mem = adjust_address (cfun->eh->sjlj_fc, TYPE_MODE (integer_type_node),
1927                             sjlj_fc_call_site_ofs);
1928       emit_move_insn (mem, GEN_INT (this_call_site));
1929       p = get_insns ();
1930       end_sequence ();
1931
1932       emit_insn_before (p, before);
1933       last_call_site = this_call_site;
1934     }
1935 }
1936
1937 /* Construct the SjLj_Function_Context.  */
1938
1939 static void
1940 sjlj_emit_function_enter (rtx dispatch_label)
1941 {
1942   rtx fn_begin, fc, mem, seq;
1943
1944   fc = cfun->eh->sjlj_fc;
1945
1946   start_sequence ();
1947
1948   /* We're storing this libcall's address into memory instead of
1949      calling it directly.  Thus, we must call assemble_external_libcall
1950      here, as we can not depend on emit_library_call to do it for us.  */
1951   assemble_external_libcall (eh_personality_libfunc);
1952   mem = adjust_address (fc, Pmode, sjlj_fc_personality_ofs);
1953   emit_move_insn (mem, eh_personality_libfunc);
1954
1955   mem = adjust_address (fc, Pmode, sjlj_fc_lsda_ofs);
1956   if (cfun->uses_eh_lsda)
1957     {
1958       char buf[20];
1959       rtx sym;
1960
1961       ASM_GENERATE_INTERNAL_LABEL (buf, "LLSDA", current_function_funcdef_no);
1962       sym = gen_rtx_SYMBOL_REF (Pmode, ggc_strdup (buf));
1963       SYMBOL_REF_FLAGS (sym) = SYMBOL_FLAG_LOCAL;
1964       emit_move_insn (mem, sym);
1965     }
1966   else
1967     emit_move_insn (mem, const0_rtx);
1968
1969 #ifdef DONT_USE_BUILTIN_SETJMP
1970   {
1971     rtx x, note;
1972     x = emit_library_call_value (setjmp_libfunc, NULL_RTX, LCT_RETURNS_TWICE,
1973                                  TYPE_MODE (integer_type_node), 1,
1974                                  plus_constant (XEXP (fc, 0),
1975                                                 sjlj_fc_jbuf_ofs), Pmode);
1976
1977     note = emit_note (NOTE_INSN_EXPECTED_VALUE);
1978     NOTE_EXPECTED_VALUE (note) = gen_rtx_EQ (VOIDmode, x, const0_rtx);
1979
1980     emit_cmp_and_jump_insns (x, const0_rtx, NE, 0,
1981                              TYPE_MODE (integer_type_node), 0, dispatch_label);
1982   }
1983 #else
1984   expand_builtin_setjmp_setup (plus_constant (XEXP (fc, 0), sjlj_fc_jbuf_ofs),
1985                                dispatch_label);
1986 #endif
1987
1988   emit_library_call (unwind_sjlj_register_libfunc, LCT_NORMAL, VOIDmode,
1989                      1, XEXP (fc, 0), Pmode);
1990
1991   seq = get_insns ();
1992   end_sequence ();
1993
1994   /* ??? Instead of doing this at the beginning of the function,
1995      do this in a block that is at loop level 0 and dominates all
1996      can_throw_internal instructions.  */
1997
1998   for (fn_begin = get_insns (); ; fn_begin = NEXT_INSN (fn_begin))
1999     if (NOTE_P (fn_begin)
2000         && (NOTE_LINE_NUMBER (fn_begin) == NOTE_INSN_FUNCTION_BEG
2001             || NOTE_LINE_NUMBER (fn_begin) == NOTE_INSN_BASIC_BLOCK))
2002       break;
2003   if (NOTE_LINE_NUMBER (fn_begin) == NOTE_INSN_FUNCTION_BEG)
2004     insert_insn_on_edge (seq, ENTRY_BLOCK_PTR->succ);
2005   else
2006     {
2007       rtx last = BB_END (ENTRY_BLOCK_PTR->succ->dest);
2008       for (; ; fn_begin = NEXT_INSN (fn_begin))
2009         if ((NOTE_P (fn_begin)
2010              && NOTE_LINE_NUMBER (fn_begin) == NOTE_INSN_FUNCTION_BEG)
2011             || fn_begin == last)
2012           break;
2013       emit_insn_after (seq, fn_begin);
2014     }
2015 }
2016
2017 /* Call back from expand_function_end to know where we should put
2018    the call to unwind_sjlj_unregister_libfunc if needed.  */
2019
2020 void
2021 sjlj_emit_function_exit_after (rtx after)
2022 {
2023   cfun->eh->sjlj_exit_after = after;
2024 }
2025
2026 static void
2027 sjlj_emit_function_exit (void)
2028 {
2029   rtx seq;
2030   edge e;
2031
2032   start_sequence ();
2033
2034   emit_library_call (unwind_sjlj_unregister_libfunc, LCT_NORMAL, VOIDmode,
2035                      1, XEXP (cfun->eh->sjlj_fc, 0), Pmode);
2036
2037   seq = get_insns ();
2038   end_sequence ();
2039
2040   /* ??? Really this can be done in any block at loop level 0 that
2041      post-dominates all can_throw_internal instructions.  This is
2042      the last possible moment.  */
2043
2044   for (e = EXIT_BLOCK_PTR->pred; e; e = e->pred_next)
2045     if (e->flags & EDGE_FALLTHRU)
2046       break;
2047   if (e)
2048     {
2049       rtx insn;
2050
2051       /* Figure out whether the place we are supposed to insert libcall
2052          is inside the last basic block or after it.  In the other case
2053          we need to emit to edge.  */
2054       if (e->src->next_bb != EXIT_BLOCK_PTR)
2055         abort ();
2056       for (insn = NEXT_INSN (BB_END (e->src)); insn; insn = NEXT_INSN (insn))
2057         if (insn == cfun->eh->sjlj_exit_after)
2058           break;
2059       if (insn)
2060         insert_insn_on_edge (seq, e);
2061       else
2062         {
2063           insn = cfun->eh->sjlj_exit_after;
2064           if (LABEL_P (insn))
2065             insn = NEXT_INSN (insn);
2066           emit_insn_after (seq, insn);
2067         }
2068     }
2069 }
2070
2071 static void
2072 sjlj_emit_dispatch_table (rtx dispatch_label, struct sjlj_lp_info *lp_info)
2073 {
2074   int i, first_reachable;
2075   rtx mem, dispatch, seq, fc;
2076   rtx before;
2077   basic_block bb;
2078   edge e;
2079
2080   fc = cfun->eh->sjlj_fc;
2081
2082   start_sequence ();
2083
2084   emit_label (dispatch_label);
2085
2086 #ifndef DONT_USE_BUILTIN_SETJMP
2087   expand_builtin_setjmp_receiver (dispatch_label);
2088 #endif
2089
2090   /* Load up dispatch index, exc_ptr and filter values from the
2091      function context.  */
2092   mem = adjust_address (fc, TYPE_MODE (integer_type_node),
2093                         sjlj_fc_call_site_ofs);
2094   dispatch = copy_to_reg (mem);
2095
2096   mem = adjust_address (fc, word_mode, sjlj_fc_data_ofs);
2097   if (word_mode != ptr_mode)
2098     {
2099 #ifdef POINTERS_EXTEND_UNSIGNED
2100       mem = convert_memory_address (ptr_mode, mem);
2101 #else
2102       mem = convert_to_mode (ptr_mode, mem, 0);
2103 #endif
2104     }
2105   emit_move_insn (cfun->eh->exc_ptr, mem);
2106
2107   mem = adjust_address (fc, word_mode, sjlj_fc_data_ofs + UNITS_PER_WORD);
2108   emit_move_insn (cfun->eh->filter, mem);
2109
2110   /* Jump to one of the directly reachable regions.  */
2111   /* ??? This really ought to be using a switch statement.  */
2112
2113   first_reachable = 0;
2114   for (i = cfun->eh->last_region_number; i > 0; --i)
2115     {
2116       if (! lp_info[i].directly_reachable)
2117         continue;
2118
2119       if (! first_reachable)
2120         {
2121           first_reachable = i;
2122           continue;
2123         }
2124
2125       emit_cmp_and_jump_insns (dispatch, GEN_INT (lp_info[i].dispatch_index),
2126                                EQ, NULL_RTX, TYPE_MODE (integer_type_node), 0,
2127                                cfun->eh->region_array[i]->post_landing_pad);
2128     }
2129
2130   seq = get_insns ();
2131   end_sequence ();
2132
2133   before = cfun->eh->region_array[first_reachable]->post_landing_pad;
2134
2135   bb = emit_to_new_bb_before (seq, before);
2136   e = make_edge (bb, bb->next_bb, EDGE_FALLTHRU);
2137   e->count = bb->count;
2138   e->probability = REG_BR_PROB_BASE;
2139 }
2140
2141 static void
2142 sjlj_build_landing_pads (void)
2143 {
2144   struct sjlj_lp_info *lp_info;
2145
2146   lp_info = xcalloc (cfun->eh->last_region_number + 1,
2147                      sizeof (struct sjlj_lp_info));
2148
2149   if (sjlj_find_directly_reachable_regions (lp_info))
2150     {
2151       rtx dispatch_label = gen_label_rtx ();
2152
2153       cfun->eh->sjlj_fc
2154         = assign_stack_local (TYPE_MODE (sjlj_fc_type_node),
2155                               int_size_in_bytes (sjlj_fc_type_node),
2156                               TYPE_ALIGN (sjlj_fc_type_node));
2157
2158       sjlj_assign_call_site_values (dispatch_label, lp_info);
2159       sjlj_mark_call_sites (lp_info);
2160
2161       sjlj_emit_function_enter (dispatch_label);
2162       sjlj_emit_dispatch_table (dispatch_label, lp_info);
2163       sjlj_emit_function_exit ();
2164     }
2165
2166   free (lp_info);
2167 }
2168
2169 void
2170 finish_eh_generation (void)
2171 {
2172   basic_block bb;
2173
2174   /* Nothing to do if no regions created.  */
2175   if (cfun->eh->region_tree == NULL)
2176     return;
2177
2178   /* The object here is to provide find_basic_blocks with detailed
2179      information (via reachable_handlers) on how exception control
2180      flows within the function.  In this first pass, we can include
2181      type information garnered from ERT_THROW and ERT_ALLOWED_EXCEPTIONS
2182      regions, and hope that it will be useful in deleting unreachable
2183      handlers.  Subsequently, we will generate landing pads which will
2184      connect many of the handlers, and then type information will not
2185      be effective.  Still, this is a win over previous implementations.  */
2186
2187   /* These registers are used by the landing pads.  Make sure they
2188      have been generated.  */
2189   get_exception_pointer (cfun);
2190   get_exception_filter (cfun);
2191
2192   /* Construct the landing pads.  */
2193
2194   assign_filter_values ();
2195   build_post_landing_pads ();
2196   connect_post_landing_pads ();
2197   if (USING_SJLJ_EXCEPTIONS)
2198     sjlj_build_landing_pads ();
2199   else
2200     dw2_build_landing_pads ();
2201
2202   cfun->eh->built_landing_pads = 1;
2203
2204   /* We've totally changed the CFG.  Start over.  */
2205   find_exception_handler_labels ();
2206   break_superblocks ();
2207   if (USING_SJLJ_EXCEPTIONS)
2208     commit_edge_insertions ();
2209   FOR_EACH_BB (bb)
2210     {
2211       edge e, next;
2212       bool eh = false;
2213       for (e = bb->succ; e; e = next)
2214         {
2215           next = e->succ_next;
2216           if (e->flags & EDGE_EH)
2217             {
2218               remove_edge (e);
2219               eh = true;
2220             }
2221         }
2222       if (eh)
2223         rtl_make_eh_edge (NULL, bb, BB_END (bb));
2224     }
2225 }
2226 \f
2227 static hashval_t
2228 ehl_hash (const void *pentry)
2229 {
2230   struct ehl_map_entry *entry = (struct ehl_map_entry *) pentry;
2231
2232   /* 2^32 * ((sqrt(5) - 1) / 2) */
2233   const hashval_t scaled_golden_ratio = 0x9e3779b9;
2234   return CODE_LABEL_NUMBER (entry->label) * scaled_golden_ratio;
2235 }
2236
2237 static int
2238 ehl_eq (const void *pentry, const void *pdata)
2239 {
2240   struct ehl_map_entry *entry = (struct ehl_map_entry *) pentry;
2241   struct ehl_map_entry *data = (struct ehl_map_entry *) pdata;
2242
2243   return entry->label == data->label;
2244 }
2245
2246 /* This section handles removing dead code for flow.  */
2247
2248 /* Remove LABEL from exception_handler_label_map.  */
2249
2250 static void
2251 remove_exception_handler_label (rtx label)
2252 {
2253   struct ehl_map_entry **slot, tmp;
2254
2255   /* If exception_handler_label_map was not built yet,
2256      there is nothing to do.  */
2257   if (cfun->eh->exception_handler_label_map == NULL)
2258     return;
2259
2260   tmp.label = label;
2261   slot = (struct ehl_map_entry **)
2262     htab_find_slot (cfun->eh->exception_handler_label_map, &tmp, NO_INSERT);
2263   if (! slot)
2264     abort ();
2265
2266   htab_clear_slot (cfun->eh->exception_handler_label_map, (void **) slot);
2267 }
2268
2269 /* Splice REGION from the region tree etc.  */
2270
2271 static void
2272 remove_eh_handler (struct eh_region *region)
2273 {
2274   struct eh_region **pp, **pp_start, *p, *outer, *inner;
2275   rtx lab;
2276
2277   /* For the benefit of efficiently handling REG_EH_REGION notes,
2278      replace this region in the region array with its containing
2279      region.  Note that previous region deletions may result in
2280      multiple copies of this region in the array, so we have a
2281      list of alternate numbers by which we are known.  */
2282
2283   outer = region->outer;
2284   cfun->eh->region_array[region->region_number] = outer;
2285   if (region->aka)
2286     {
2287       int i;
2288       EXECUTE_IF_SET_IN_BITMAP (region->aka, 0, i,
2289         { cfun->eh->region_array[i] = outer; });
2290     }
2291
2292   if (outer)
2293     {
2294       if (!outer->aka)
2295         outer->aka = BITMAP_GGC_ALLOC ();
2296       if (region->aka)
2297         bitmap_a_or_b (outer->aka, outer->aka, region->aka);
2298       bitmap_set_bit (outer->aka, region->region_number);
2299     }
2300
2301   if (cfun->eh->built_landing_pads)
2302     lab = region->landing_pad;
2303   else
2304     lab = region->label;
2305   if (lab)
2306     remove_exception_handler_label (lab);
2307
2308   if (outer)
2309     pp_start = &outer->inner;
2310   else
2311     pp_start = &cfun->eh->region_tree;
2312   for (pp = pp_start, p = *pp; p != region; pp = &p->next_peer, p = *pp)
2313     continue;
2314   *pp = region->next_peer;
2315
2316   inner = region->inner;
2317   if (inner)
2318     {
2319       for (p = inner; p->next_peer ; p = p->next_peer)
2320         p->outer = outer;
2321       p->outer = outer;
2322
2323       p->next_peer = *pp_start;
2324       *pp_start = inner;
2325     }
2326
2327   if (region->type == ERT_CATCH)
2328     {
2329       struct eh_region *try, *next, *prev;
2330
2331       for (try = region->next_peer;
2332            try->type == ERT_CATCH;
2333            try = try->next_peer)
2334         continue;
2335       if (try->type != ERT_TRY)
2336         abort ();
2337
2338       next = region->u.catch.next_catch;
2339       prev = region->u.catch.prev_catch;
2340
2341       if (next)
2342         next->u.catch.prev_catch = prev;
2343       else
2344         try->u.try.last_catch = prev;
2345       if (prev)
2346         prev->u.catch.next_catch = next;
2347       else
2348         {
2349           try->u.try.catch = next;
2350           if (! next)
2351             remove_eh_handler (try);
2352         }
2353     }
2354 }
2355
2356 /* LABEL heads a basic block that is about to be deleted.  If this
2357    label corresponds to an exception region, we may be able to
2358    delete the region.  */
2359
2360 void
2361 maybe_remove_eh_handler (rtx label)
2362 {
2363   struct ehl_map_entry **slot, tmp;
2364   struct eh_region *region;
2365
2366   /* ??? After generating landing pads, it's not so simple to determine
2367      if the region data is completely unused.  One must examine the
2368      landing pad and the post landing pad, and whether an inner try block
2369      is referencing the catch handlers directly.  */
2370   if (cfun->eh->built_landing_pads)
2371     return;
2372
2373   tmp.label = label;
2374   slot = (struct ehl_map_entry **)
2375     htab_find_slot (cfun->eh->exception_handler_label_map, &tmp, NO_INSERT);
2376   if (! slot)
2377     return;
2378   region = (*slot)->region;
2379   if (! region)
2380     return;
2381
2382   /* Flow will want to remove MUST_NOT_THROW regions as unreachable
2383      because there is no path to the fallback call to terminate.
2384      But the region continues to affect call-site data until there
2385      are no more contained calls, which we don't see here.  */
2386   if (region->type == ERT_MUST_NOT_THROW)
2387     {
2388       htab_clear_slot (cfun->eh->exception_handler_label_map, (void **) slot);
2389       region->label = NULL_RTX;
2390     }
2391   else
2392     remove_eh_handler (region);
2393 }
2394
2395 /* Invokes CALLBACK for every exception handler label.  Only used by old
2396    loop hackery; should not be used by new code.  */
2397
2398 void
2399 for_each_eh_label (void (*callback) (rtx))
2400 {
2401   htab_traverse (cfun->eh->exception_handler_label_map, for_each_eh_label_1,
2402                  (void *) &callback);
2403 }
2404
2405 static int
2406 for_each_eh_label_1 (void **pentry, void *data)
2407 {
2408   struct ehl_map_entry *entry = *(struct ehl_map_entry **)pentry;
2409   void (*callback) (rtx) = *(void (**) (rtx)) data;
2410
2411   (*callback) (entry->label);
2412   return 1;
2413 }
2414
2415 /* Invoke CALLBACK for every exception region in the current function.  */
2416
2417 void
2418 for_each_eh_region (void (*callback) (struct eh_region *))
2419 {
2420   int i, n = cfun->eh->last_region_number;
2421   for (i = 1; i <= n; ++i)
2422     {
2423       struct eh_region *region = cfun->eh->region_array[i];
2424       if (region)
2425         (*callback) (region);
2426     }
2427 }
2428 \f
2429 /* This section describes CFG exception edges for flow.  */
2430
2431 /* For communicating between calls to reachable_next_level.  */
2432 struct reachable_info
2433 {
2434   tree types_caught;
2435   tree types_allowed;
2436   void (*callback) (struct eh_region *, void *);
2437   void *callback_data;
2438   bool saw_any_handlers;
2439 };
2440
2441 /* A subroutine of reachable_next_level.  Return true if TYPE, or a
2442    base class of TYPE, is in HANDLED.  */
2443
2444 int
2445 check_handled (tree handled, tree type)
2446 {
2447   tree t;
2448
2449   /* We can check for exact matches without front-end help.  */
2450   if (! lang_eh_type_covers)
2451     {
2452       for (t = handled; t ; t = TREE_CHAIN (t))
2453         if (TREE_VALUE (t) == type)
2454           return 1;
2455     }
2456   else
2457     {
2458       for (t = handled; t ; t = TREE_CHAIN (t))
2459         if ((*lang_eh_type_covers) (TREE_VALUE (t), type))
2460           return 1;
2461     }
2462
2463   return 0;
2464 }
2465
2466 /* A subroutine of reachable_next_level.  If we are collecting a list
2467    of handlers, add one.  After landing pad generation, reference
2468    it instead of the handlers themselves.  Further, the handlers are
2469    all wired together, so by referencing one, we've got them all.
2470    Before landing pad generation we reference each handler individually.
2471
2472    LP_REGION contains the landing pad; REGION is the handler.  */
2473
2474 static void
2475 add_reachable_handler (struct reachable_info *info,
2476                        struct eh_region *lp_region, struct eh_region *region)
2477 {
2478   if (! info)
2479     return;
2480
2481   info->saw_any_handlers = true;
2482
2483   if (cfun->eh->built_landing_pads)
2484     info->callback (lp_region, info->callback_data);
2485   else
2486     info->callback (region, info->callback_data);
2487 }
2488
2489 /* Process one level of exception regions for reachability.
2490    If TYPE_THROWN is non-null, then it is the *exact* type being
2491    propagated.  If INFO is non-null, then collect handler labels
2492    and caught/allowed type information between invocations.  */
2493
2494 static enum reachable_code
2495 reachable_next_level (struct eh_region *region, tree type_thrown,
2496                       struct reachable_info *info)
2497 {
2498   switch (region->type)
2499     {
2500     case ERT_CLEANUP:
2501       /* Before landing-pad generation, we model control flow
2502          directly to the individual handlers.  In this way we can
2503          see that catch handler types may shadow one another.  */
2504       add_reachable_handler (info, region, region);
2505       return RNL_MAYBE_CAUGHT;
2506
2507     case ERT_TRY:
2508       {
2509         struct eh_region *c;
2510         enum reachable_code ret = RNL_NOT_CAUGHT;
2511
2512         for (c = region->u.try.catch; c ; c = c->u.catch.next_catch)
2513           {
2514             /* A catch-all handler ends the search.  */
2515             if (c->u.catch.type_list == NULL)
2516               {
2517                 add_reachable_handler (info, region, c);
2518                 return RNL_CAUGHT;
2519               }
2520
2521             if (type_thrown)
2522               {
2523                 /* If we have at least one type match, end the search.  */
2524                 tree tp_node = c->u.catch.type_list;
2525
2526                 for (; tp_node; tp_node = TREE_CHAIN (tp_node))
2527                   {
2528                     tree type = TREE_VALUE (tp_node);
2529
2530                     if (type == type_thrown
2531                         || (lang_eh_type_covers
2532                             && (*lang_eh_type_covers) (type, type_thrown)))
2533                       {
2534                         add_reachable_handler (info, region, c);
2535                         return RNL_CAUGHT;
2536                       }
2537                   }
2538
2539                 /* If we have definitive information of a match failure,
2540                    the catch won't trigger.  */
2541                 if (lang_eh_type_covers)
2542                   return RNL_NOT_CAUGHT;
2543               }
2544
2545             /* At this point, we either don't know what type is thrown or
2546                don't have front-end assistance to help deciding if it is
2547                covered by one of the types in the list for this region.
2548
2549                We'd then like to add this region to the list of reachable
2550                handlers since it is indeed potentially reachable based on the
2551                information we have.
2552
2553                Actually, this handler is for sure not reachable if all the
2554                types it matches have already been caught. That is, it is only
2555                potentially reachable if at least one of the types it catches
2556                has not been previously caught.  */
2557
2558             if (! info)
2559               ret = RNL_MAYBE_CAUGHT;
2560             else
2561               {
2562                 tree tp_node = c->u.catch.type_list;
2563                 bool maybe_reachable = false;
2564
2565                 /* Compute the potential reachability of this handler and
2566                    update the list of types caught at the same time.  */
2567                 for (; tp_node; tp_node = TREE_CHAIN (tp_node))
2568                   {
2569                     tree type = TREE_VALUE (tp_node);
2570
2571                     if (! check_handled (info->types_caught, type))
2572                       {
2573                         info->types_caught
2574                           = tree_cons (NULL, type, info->types_caught);
2575
2576                         maybe_reachable = true;
2577                       }
2578                   }
2579
2580                 if (maybe_reachable)
2581                   {
2582                     add_reachable_handler (info, region, c);
2583
2584                     /* ??? If the catch type is a base class of every allowed
2585                        type, then we know we can stop the search.  */
2586                     ret = RNL_MAYBE_CAUGHT;
2587                   }
2588               }
2589           }
2590
2591         return ret;
2592       }
2593
2594     case ERT_ALLOWED_EXCEPTIONS:
2595       /* An empty list of types definitely ends the search.  */
2596       if (region->u.allowed.type_list == NULL_TREE)
2597         {
2598           add_reachable_handler (info, region, region);
2599           return RNL_CAUGHT;
2600         }
2601
2602       /* Collect a list of lists of allowed types for use in detecting
2603          when a catch may be transformed into a catch-all.  */
2604       if (info)
2605         info->types_allowed = tree_cons (NULL_TREE,
2606                                          region->u.allowed.type_list,
2607                                          info->types_allowed);
2608
2609       /* If we have definitive information about the type hierarchy,
2610          then we can tell if the thrown type will pass through the
2611          filter.  */
2612       if (type_thrown && lang_eh_type_covers)
2613         {
2614           if (check_handled (region->u.allowed.type_list, type_thrown))
2615             return RNL_NOT_CAUGHT;
2616           else
2617             {
2618               add_reachable_handler (info, region, region);
2619               return RNL_CAUGHT;
2620             }
2621         }
2622
2623       add_reachable_handler (info, region, region);
2624       return RNL_MAYBE_CAUGHT;
2625
2626     case ERT_CATCH:
2627       /* Catch regions are handled by their controlling try region.  */
2628       return RNL_NOT_CAUGHT;
2629
2630     case ERT_MUST_NOT_THROW:
2631       /* Here we end our search, since no exceptions may propagate.
2632          If we've touched down at some landing pad previous, then the
2633          explicit function call we generated may be used.  Otherwise
2634          the call is made by the runtime.  */
2635       if (info && info->saw_any_handlers)
2636         {
2637           add_reachable_handler (info, region, region);
2638           return RNL_CAUGHT;
2639         }
2640       else
2641         return RNL_BLOCKED;
2642
2643     case ERT_THROW:
2644     case ERT_FIXUP:
2645     case ERT_UNKNOWN:
2646       /* Shouldn't see these here.  */
2647       break;
2648     }
2649
2650   abort ();
2651 }
2652
2653 /* Invoke CALLBACK on each region reachable from REGION_NUMBER.  */
2654
2655 void
2656 foreach_reachable_handler (int region_number, bool is_resx,
2657                            void (*callback) (struct eh_region *, void *),
2658                            void *callback_data)
2659 {
2660   struct reachable_info info;
2661   struct eh_region *region;
2662   tree type_thrown;
2663
2664   memset (&info, 0, sizeof (info));
2665   info.callback = callback;
2666   info.callback_data = callback_data;
2667
2668   region = cfun->eh->region_array[region_number];
2669
2670   type_thrown = NULL_TREE;
2671   if (is_resx)
2672     {
2673       /* A RESX leaves a region instead of entering it.  Thus the
2674          region itself may have been deleted out from under us.  */
2675       if (region == NULL)
2676         return;
2677       region = region->outer;
2678     }
2679   else if (region->type == ERT_THROW)
2680     {
2681       type_thrown = region->u.throw.type;
2682       region = region->outer;
2683     }
2684
2685   while (region)
2686     {
2687       if (reachable_next_level (region, type_thrown, &info) >= RNL_CAUGHT)
2688         break;
2689       /* If we have processed one cleanup, there is no point in
2690          processing any more of them.  Each cleanup will have an edge
2691          to the next outer cleanup region, so the flow graph will be
2692          accurate.  */
2693       if (region->type == ERT_CLEANUP)
2694         region = region->u.cleanup.prev_try;
2695       else
2696         region = region->outer;
2697     }
2698 }
2699
2700 /* Retrieve a list of labels of exception handlers which can be
2701    reached by a given insn.  */
2702
2703 static void
2704 arh_to_landing_pad (struct eh_region *region, void *data)
2705 {
2706   rtx *p_handlers = data;
2707   if (! *p_handlers)
2708     *p_handlers = alloc_INSN_LIST (region->landing_pad, NULL_RTX);
2709 }
2710
2711 static void
2712 arh_to_label (struct eh_region *region, void *data)
2713 {
2714   rtx *p_handlers = data;
2715   *p_handlers = alloc_INSN_LIST (region->label, *p_handlers);
2716 }
2717
2718 rtx
2719 reachable_handlers (rtx insn)
2720 {
2721   bool is_resx = false;
2722   rtx handlers = NULL;
2723   int region_number;
2724
2725   if (JUMP_P (insn)
2726       && GET_CODE (PATTERN (insn)) == RESX)
2727     {
2728       region_number = XINT (PATTERN (insn), 0);
2729       is_resx = true;
2730     }
2731   else
2732     {
2733       rtx note = find_reg_note (insn, REG_EH_REGION, NULL_RTX);
2734       if (!note || INTVAL (XEXP (note, 0)) <= 0)
2735         return NULL;
2736       region_number = INTVAL (XEXP (note, 0));
2737     }
2738
2739   foreach_reachable_handler (region_number, is_resx,
2740                              (cfun->eh->built_landing_pads
2741                               ? arh_to_landing_pad
2742                               : arh_to_label),
2743                              &handlers);
2744
2745   return handlers;
2746 }
2747
2748 /* Determine if the given INSN can throw an exception that is caught
2749    within the function.  */
2750
2751 bool
2752 can_throw_internal_1 (int region_number)
2753 {
2754   struct eh_region *region;
2755   tree type_thrown;
2756
2757   region = cfun->eh->region_array[region_number];
2758
2759   type_thrown = NULL_TREE;
2760   if (region->type == ERT_THROW)
2761     {
2762       type_thrown = region->u.throw.type;
2763       region = region->outer;
2764     }
2765
2766   /* If this exception is ignored by each and every containing region,
2767      then control passes straight out.  The runtime may handle some
2768      regions, which also do not require processing internally.  */
2769   for (; region; region = region->outer)
2770     {
2771       enum reachable_code how = reachable_next_level (region, type_thrown, 0);
2772       if (how == RNL_BLOCKED)
2773         return false;
2774       if (how != RNL_NOT_CAUGHT)
2775         return true;
2776     }
2777
2778   return false;
2779 }
2780
2781 bool
2782 can_throw_internal (rtx insn)
2783 {
2784   rtx note;
2785
2786   if (! INSN_P (insn))
2787     return false;
2788
2789   if (JUMP_P (insn)
2790       && GET_CODE (PATTERN (insn)) == RESX
2791       && XINT (PATTERN (insn), 0) > 0)
2792     return can_throw_internal_1 (XINT (PATTERN (insn), 0));
2793
2794   if (NONJUMP_INSN_P (insn)
2795       && GET_CODE (PATTERN (insn)) == SEQUENCE)
2796     insn = XVECEXP (PATTERN (insn), 0, 0);
2797
2798   /* Every insn that might throw has an EH_REGION note.  */
2799   note = find_reg_note (insn, REG_EH_REGION, NULL_RTX);
2800   if (!note || INTVAL (XEXP (note, 0)) <= 0)
2801     return false;
2802
2803   return can_throw_internal_1 (INTVAL (XEXP (note, 0)));
2804 }
2805
2806 /* Determine if the given INSN can throw an exception that is
2807    visible outside the function.  */
2808
2809 bool
2810 can_throw_external_1 (int region_number)
2811 {
2812   struct eh_region *region;
2813   tree type_thrown;
2814
2815   region = cfun->eh->region_array[region_number];
2816
2817   type_thrown = NULL_TREE;
2818   if (region->type == ERT_THROW)
2819     {
2820       type_thrown = region->u.throw.type;
2821       region = region->outer;
2822     }
2823
2824   /* If the exception is caught or blocked by any containing region,
2825      then it is not seen by any calling function.  */
2826   for (; region ; region = region->outer)
2827     if (reachable_next_level (region, type_thrown, NULL) >= RNL_CAUGHT)
2828       return false;
2829
2830   return true;
2831 }
2832
2833 bool
2834 can_throw_external (rtx insn)
2835 {
2836   rtx note;
2837
2838   if (! INSN_P (insn))
2839     return false;
2840
2841   if (NONJUMP_INSN_P (insn)
2842       && GET_CODE (PATTERN (insn)) == SEQUENCE)
2843     insn = XVECEXP (PATTERN (insn), 0, 0);
2844
2845   note = find_reg_note (insn, REG_EH_REGION, NULL_RTX);
2846   if (!note)
2847     {
2848       /* Calls (and trapping insns) without notes are outside any
2849          exception handling region in this function.  We have to
2850          assume it might throw.  Given that the front end and middle
2851          ends mark known NOTHROW functions, this isn't so wildly
2852          inaccurate.  */
2853       return (CALL_P (insn)
2854               || (flag_non_call_exceptions
2855                   && may_trap_p (PATTERN (insn))));
2856     }
2857   if (INTVAL (XEXP (note, 0)) <= 0)
2858     return false;
2859
2860   return can_throw_external_1 (INTVAL (XEXP (note, 0)));
2861 }
2862
2863 /* Set TREE_NOTHROW and cfun->all_throwers_are_sibcalls.  */
2864
2865 void
2866 set_nothrow_function_flags (void)
2867 {
2868   rtx insn;
2869
2870   TREE_NOTHROW (current_function_decl) = 1;
2871
2872   /* Assume cfun->all_throwers_are_sibcalls until we encounter
2873      something that can throw an exception.  We specifically exempt
2874      CALL_INSNs that are SIBLING_CALL_P, as these are really jumps,
2875      and can't throw.  Most CALL_INSNs are not SIBLING_CALL_P, so this
2876      is optimistic.  */
2877
2878   cfun->all_throwers_are_sibcalls = 1;
2879
2880   if (! flag_exceptions)
2881     return;
2882
2883   for (insn = get_insns (); insn; insn = NEXT_INSN (insn))
2884     if (can_throw_external (insn))
2885       {
2886         TREE_NOTHROW (current_function_decl) = 0;
2887
2888         if (!CALL_P (insn) || !SIBLING_CALL_P (insn))
2889           {
2890             cfun->all_throwers_are_sibcalls = 0;
2891             return;
2892           }
2893       }
2894
2895   for (insn = current_function_epilogue_delay_list; insn;
2896        insn = XEXP (insn, 1))
2897     if (can_throw_external (insn))
2898       {
2899         TREE_NOTHROW (current_function_decl) = 0;
2900
2901         if (!CALL_P (insn) || !SIBLING_CALL_P (insn))
2902           {
2903             cfun->all_throwers_are_sibcalls = 0;
2904             return;
2905           }
2906       }
2907 }
2908
2909 \f
2910 /* Various hooks for unwind library.  */
2911
2912 /* Do any necessary initialization to access arbitrary stack frames.
2913    On the SPARC, this means flushing the register windows.  */
2914
2915 void
2916 expand_builtin_unwind_init (void)
2917 {
2918   /* Set this so all the registers get saved in our frame; we need to be
2919      able to copy the saved values for any registers from frames we unwind.  */
2920   current_function_has_nonlocal_label = 1;
2921
2922 #ifdef SETUP_FRAME_ADDRESSES
2923   SETUP_FRAME_ADDRESSES ();
2924 #endif
2925 }
2926
2927 rtx
2928 expand_builtin_eh_return_data_regno (tree arglist)
2929 {
2930   tree which = TREE_VALUE (arglist);
2931   unsigned HOST_WIDE_INT iwhich;
2932
2933   if (TREE_CODE (which) != INTEGER_CST)
2934     {
2935       error ("argument of `__builtin_eh_return_regno' must be constant");
2936       return constm1_rtx;
2937     }
2938
2939   iwhich = tree_low_cst (which, 1);
2940   iwhich = EH_RETURN_DATA_REGNO (iwhich);
2941   if (iwhich == INVALID_REGNUM)
2942     return constm1_rtx;
2943
2944 #ifdef DWARF_FRAME_REGNUM
2945   iwhich = DWARF_FRAME_REGNUM (iwhich);
2946 #else
2947   iwhich = DBX_REGISTER_NUMBER (iwhich);
2948 #endif
2949
2950   return GEN_INT (iwhich);
2951 }
2952
2953 /* Given a value extracted from the return address register or stack slot,
2954    return the actual address encoded in that value.  */
2955
2956 rtx
2957 expand_builtin_extract_return_addr (tree addr_tree)
2958 {
2959   rtx addr = expand_expr (addr_tree, NULL_RTX, Pmode, 0);
2960
2961   if (GET_MODE (addr) != Pmode
2962       && GET_MODE (addr) != VOIDmode)
2963     {
2964 #ifdef POINTERS_EXTEND_UNSIGNED
2965       addr = convert_memory_address (Pmode, addr);
2966 #else
2967       addr = convert_to_mode (Pmode, addr, 0);
2968 #endif
2969     }
2970
2971   /* First mask out any unwanted bits.  */
2972 #ifdef MASK_RETURN_ADDR
2973   expand_and (Pmode, addr, MASK_RETURN_ADDR, addr);
2974 #endif
2975
2976   /* Then adjust to find the real return address.  */
2977 #if defined (RETURN_ADDR_OFFSET)
2978   addr = plus_constant (addr, RETURN_ADDR_OFFSET);
2979 #endif
2980
2981   return addr;
2982 }
2983
2984 /* Given an actual address in addr_tree, do any necessary encoding
2985    and return the value to be stored in the return address register or
2986    stack slot so the epilogue will return to that address.  */
2987
2988 rtx
2989 expand_builtin_frob_return_addr (tree addr_tree)
2990 {
2991   rtx addr = expand_expr (addr_tree, NULL_RTX, ptr_mode, 0);
2992
2993   addr = convert_memory_address (Pmode, addr);
2994
2995 #ifdef RETURN_ADDR_OFFSET
2996   addr = force_reg (Pmode, addr);
2997   addr = plus_constant (addr, -RETURN_ADDR_OFFSET);
2998 #endif
2999
3000   return addr;
3001 }
3002
3003 /* Set up the epilogue with the magic bits we'll need to return to the
3004    exception handler.  */
3005
3006 void
3007 expand_builtin_eh_return (tree stackadj_tree ATTRIBUTE_UNUSED,
3008                           tree handler_tree)
3009 {
3010   rtx tmp;
3011
3012 #ifdef EH_RETURN_STACKADJ_RTX
3013   tmp = expand_expr (stackadj_tree, cfun->eh->ehr_stackadj, VOIDmode, 0);
3014   tmp = convert_memory_address (Pmode, tmp);
3015   if (!cfun->eh->ehr_stackadj)
3016     cfun->eh->ehr_stackadj = copy_to_reg (tmp);
3017   else if (tmp != cfun->eh->ehr_stackadj)
3018     emit_move_insn (cfun->eh->ehr_stackadj, tmp);
3019 #endif
3020
3021   tmp = expand_expr (handler_tree, cfun->eh->ehr_handler, VOIDmode, 0);
3022   tmp = convert_memory_address (Pmode, tmp);
3023   if (!cfun->eh->ehr_handler)
3024     cfun->eh->ehr_handler = copy_to_reg (tmp);
3025   else if (tmp != cfun->eh->ehr_handler)
3026     emit_move_insn (cfun->eh->ehr_handler, tmp);
3027
3028   if (!cfun->eh->ehr_label)
3029     cfun->eh->ehr_label = gen_label_rtx ();
3030   emit_jump (cfun->eh->ehr_label);
3031 }
3032
3033 void
3034 expand_eh_return (void)
3035 {
3036   rtx around_label;
3037
3038   if (! cfun->eh->ehr_label)
3039     return;
3040
3041   current_function_calls_eh_return = 1;
3042
3043 #ifdef EH_RETURN_STACKADJ_RTX
3044   emit_move_insn (EH_RETURN_STACKADJ_RTX, const0_rtx);
3045 #endif
3046
3047   around_label = gen_label_rtx ();
3048   emit_jump (around_label);
3049
3050   emit_label (cfun->eh->ehr_label);
3051   clobber_return_register ();
3052
3053 #ifdef EH_RETURN_STACKADJ_RTX
3054   emit_move_insn (EH_RETURN_STACKADJ_RTX, cfun->eh->ehr_stackadj);
3055 #endif
3056
3057 #ifdef HAVE_eh_return
3058   if (HAVE_eh_return)
3059     emit_insn (gen_eh_return (cfun->eh->ehr_handler));
3060   else
3061 #endif
3062     {
3063 #ifdef EH_RETURN_HANDLER_RTX
3064       emit_move_insn (EH_RETURN_HANDLER_RTX, cfun->eh->ehr_handler);
3065 #else
3066       error ("__builtin_eh_return not supported on this target");
3067 #endif
3068     }
3069
3070   emit_label (around_label);
3071 }
3072
3073 /* Convert a ptr_mode address ADDR_TREE to a Pmode address controlled by
3074    POINTERS_EXTEND_UNSIGNED and return it.  */
3075
3076 rtx
3077 expand_builtin_extend_pointer (tree addr_tree)
3078 {
3079   rtx addr = expand_expr (addr_tree, NULL_RTX, ptr_mode, 0);
3080   int extend;
3081
3082 #ifdef POINTERS_EXTEND_UNSIGNED
3083   extend = POINTERS_EXTEND_UNSIGNED;
3084 #else
3085   /* The previous EH code did an unsigned extend by default, so we do this also
3086      for consistency.  */
3087   extend = 1;
3088 #endif
3089
3090   return convert_modes (word_mode, ptr_mode, addr, extend);
3091 }
3092 \f
3093 /* In the following functions, we represent entries in the action table
3094    as 1-based indices.  Special cases are:
3095
3096          0:     null action record, non-null landing pad; implies cleanups
3097         -1:     null action record, null landing pad; implies no action
3098         -2:     no call-site entry; implies must_not_throw
3099         -3:     we have yet to process outer regions
3100
3101    Further, no special cases apply to the "next" field of the record.
3102    For next, 0 means end of list.  */
3103
3104 struct action_record
3105 {
3106   int offset;
3107   int filter;
3108   int next;
3109 };
3110
3111 static int
3112 action_record_eq (const void *pentry, const void *pdata)
3113 {
3114   const struct action_record *entry = (const struct action_record *) pentry;
3115   const struct action_record *data = (const struct action_record *) pdata;
3116   return entry->filter == data->filter && entry->next == data->next;
3117 }
3118
3119 static hashval_t
3120 action_record_hash (const void *pentry)
3121 {
3122   const struct action_record *entry = (const struct action_record *) pentry;
3123   return entry->next * 1009 + entry->filter;
3124 }
3125
3126 static int
3127 add_action_record (htab_t ar_hash, int filter, int next)
3128 {
3129   struct action_record **slot, *new, tmp;
3130
3131   tmp.filter = filter;
3132   tmp.next = next;
3133   slot = (struct action_record **) htab_find_slot (ar_hash, &tmp, INSERT);
3134
3135   if ((new = *slot) == NULL)
3136     {
3137       new = xmalloc (sizeof (*new));
3138       new->offset = VARRAY_ACTIVE_SIZE (cfun->eh->action_record_data) + 1;
3139       new->filter = filter;
3140       new->next = next;
3141       *slot = new;
3142
3143       /* The filter value goes in untouched.  The link to the next
3144          record is a "self-relative" byte offset, or zero to indicate
3145          that there is no next record.  So convert the absolute 1 based
3146          indices we've been carrying around into a displacement.  */
3147
3148       push_sleb128 (&cfun->eh->action_record_data, filter);
3149       if (next)
3150         next -= VARRAY_ACTIVE_SIZE (cfun->eh->action_record_data) + 1;
3151       push_sleb128 (&cfun->eh->action_record_data, next);
3152     }
3153
3154   return new->offset;
3155 }
3156
3157 static int
3158 collect_one_action_chain (htab_t ar_hash, struct eh_region *region)
3159 {
3160   struct eh_region *c;
3161   int next;
3162
3163   /* If we've reached the top of the region chain, then we have
3164      no actions, and require no landing pad.  */
3165   if (region == NULL)
3166     return -1;
3167
3168   switch (region->type)
3169     {
3170     case ERT_CLEANUP:
3171       /* A cleanup adds a zero filter to the beginning of the chain, but
3172          there are special cases to look out for.  If there are *only*
3173          cleanups along a path, then it compresses to a zero action.
3174          Further, if there are multiple cleanups along a path, we only
3175          need to represent one of them, as that is enough to trigger
3176          entry to the landing pad at runtime.  */
3177       next = collect_one_action_chain (ar_hash, region->outer);
3178       if (next <= 0)
3179         return 0;
3180       for (c = region->outer; c ; c = c->outer)
3181         if (c->type == ERT_CLEANUP)
3182           return next;
3183       return add_action_record (ar_hash, 0, next);
3184
3185     case ERT_TRY:
3186       /* Process the associated catch regions in reverse order.
3187          If there's a catch-all handler, then we don't need to
3188          search outer regions.  Use a magic -3 value to record
3189          that we haven't done the outer search.  */
3190       next = -3;
3191       for (c = region->u.try.last_catch; c ; c = c->u.catch.prev_catch)
3192         {
3193           if (c->u.catch.type_list == NULL)
3194             {
3195               /* Retrieve the filter from the head of the filter list
3196                  where we have stored it (see assign_filter_values).  */
3197               int filter
3198                 = TREE_INT_CST_LOW (TREE_VALUE (c->u.catch.filter_list));
3199
3200               next = add_action_record (ar_hash, filter, 0);
3201             }
3202           else
3203             {
3204               /* Once the outer search is done, trigger an action record for
3205                  each filter we have.  */
3206               tree flt_node;
3207
3208               if (next == -3)
3209                 {
3210                   next = collect_one_action_chain (ar_hash, region->outer);
3211
3212                   /* If there is no next action, terminate the chain.  */
3213                   if (next == -1)
3214                     next = 0;
3215                   /* If all outer actions are cleanups or must_not_throw,
3216                      we'll have no action record for it, since we had wanted
3217                      to encode these states in the call-site record directly.
3218                      Add a cleanup action to the chain to catch these.  */
3219                   else if (next <= 0)
3220                     next = add_action_record (ar_hash, 0, 0);
3221                 }
3222
3223               flt_node = c->u.catch.filter_list;
3224               for (; flt_node; flt_node = TREE_CHAIN (flt_node))
3225                 {
3226                   int filter = TREE_INT_CST_LOW (TREE_VALUE (flt_node));
3227                   next = add_action_record (ar_hash, filter, next);
3228                 }
3229             }
3230         }
3231       return next;
3232
3233     case ERT_ALLOWED_EXCEPTIONS:
3234       /* An exception specification adds its filter to the
3235          beginning of the chain.  */
3236       next = collect_one_action_chain (ar_hash, region->outer);
3237
3238       /* If there is no next action, terminate the chain.  */
3239       if (next == -1)
3240         next = 0;
3241       /* If all outer actions are cleanups or must_not_throw,
3242          we'll have no action record for it, since we had wanted
3243          to encode these states in the call-site record directly.
3244          Add a cleanup action to the chain to catch these.  */
3245       else if (next <= 0)
3246         next = add_action_record (ar_hash, 0, 0);
3247       
3248       return add_action_record (ar_hash, region->u.allowed.filter, next);
3249
3250     case ERT_MUST_NOT_THROW:
3251       /* A must-not-throw region with no inner handlers or cleanups
3252          requires no call-site entry.  Note that this differs from
3253          the no handler or cleanup case in that we do require an lsda
3254          to be generated.  Return a magic -2 value to record this.  */
3255       return -2;
3256
3257     case ERT_CATCH:
3258     case ERT_THROW:
3259       /* CATCH regions are handled in TRY above.  THROW regions are
3260          for optimization information only and produce no output.  */
3261       return collect_one_action_chain (ar_hash, region->outer);
3262
3263     default:
3264       abort ();
3265     }
3266 }
3267
3268 static int
3269 add_call_site (rtx landing_pad, int action)
3270 {
3271   struct call_site_record *data = cfun->eh->call_site_data;
3272   int used = cfun->eh->call_site_data_used;
3273   int size = cfun->eh->call_site_data_size;
3274
3275   if (used >= size)
3276     {
3277       size = (size ? size * 2 : 64);
3278       data = ggc_realloc (data, sizeof (*data) * size);
3279       cfun->eh->call_site_data = data;
3280       cfun->eh->call_site_data_size = size;
3281     }
3282
3283   data[used].landing_pad = landing_pad;
3284   data[used].action = action;
3285
3286   cfun->eh->call_site_data_used = used + 1;
3287
3288   return used + call_site_base;
3289 }
3290
3291 /* Turn REG_EH_REGION notes back into NOTE_INSN_EH_REGION notes.
3292    The new note numbers will not refer to region numbers, but
3293    instead to call site entries.  */
3294
3295 void
3296 convert_to_eh_region_ranges (void)
3297 {
3298   rtx insn, iter, note;
3299   htab_t ar_hash;
3300   int last_action = -3;
3301   rtx last_action_insn = NULL_RTX;
3302   rtx last_landing_pad = NULL_RTX;
3303   rtx first_no_action_insn = NULL_RTX;
3304   int call_site = 0;
3305
3306   if (USING_SJLJ_EXCEPTIONS || cfun->eh->region_tree == NULL)
3307     return;
3308
3309   VARRAY_UCHAR_INIT (cfun->eh->action_record_data, 64, "action_record_data");
3310
3311   ar_hash = htab_create (31, action_record_hash, action_record_eq, free);
3312
3313   for (iter = get_insns (); iter ; iter = NEXT_INSN (iter))
3314     if (INSN_P (iter))
3315       {
3316         struct eh_region *region;
3317         int this_action;
3318         rtx this_landing_pad;
3319
3320         insn = iter;
3321         if (NONJUMP_INSN_P (insn)
3322             && GET_CODE (PATTERN (insn)) == SEQUENCE)
3323           insn = XVECEXP (PATTERN (insn), 0, 0);
3324
3325         note = find_reg_note (insn, REG_EH_REGION, NULL_RTX);
3326         if (!note)
3327           {
3328             if (! (CALL_P (insn)
3329                    || (flag_non_call_exceptions
3330                        && may_trap_p (PATTERN (insn)))))
3331               continue;
3332             this_action = -1;
3333             region = NULL;
3334           }
3335         else
3336           {
3337             if (INTVAL (XEXP (note, 0)) <= 0)
3338               continue;
3339             region = cfun->eh->region_array[INTVAL (XEXP (note, 0))];
3340             this_action = collect_one_action_chain (ar_hash, region);
3341           }
3342
3343         /* Existence of catch handlers, or must-not-throw regions
3344            implies that an lsda is needed (even if empty).  */
3345         if (this_action != -1)
3346           cfun->uses_eh_lsda = 1;
3347
3348         /* Delay creation of region notes for no-action regions
3349            until we're sure that an lsda will be required.  */
3350         else if (last_action == -3)
3351           {
3352             first_no_action_insn = iter;
3353             last_action = -1;
3354           }
3355
3356         /* Cleanups and handlers may share action chains but not
3357            landing pads.  Collect the landing pad for this region.  */
3358         if (this_action >= 0)
3359           {
3360             struct eh_region *o;
3361             for (o = region; ! o->landing_pad ; o = o->outer)
3362               continue;
3363             this_landing_pad = o->landing_pad;
3364           }
3365         else
3366           this_landing_pad = NULL_RTX;
3367
3368         /* Differing actions or landing pads implies a change in call-site
3369            info, which implies some EH_REGION note should be emitted.  */
3370         if (last_action != this_action
3371             || last_landing_pad != this_landing_pad)
3372           {
3373             /* If we'd not seen a previous action (-3) or the previous
3374                action was must-not-throw (-2), then we do not need an
3375                end note.  */
3376             if (last_action >= -1)
3377               {
3378                 /* If we delayed the creation of the begin, do it now.  */
3379                 if (first_no_action_insn)
3380                   {
3381                     call_site = add_call_site (NULL_RTX, 0);
3382                     note = emit_note_before (NOTE_INSN_EH_REGION_BEG,
3383                                              first_no_action_insn);
3384                     NOTE_EH_HANDLER (note) = call_site;
3385                     first_no_action_insn = NULL_RTX;
3386                   }
3387
3388                 note = emit_note_after (NOTE_INSN_EH_REGION_END,
3389                                         last_action_insn);
3390                 NOTE_EH_HANDLER (note) = call_site;
3391               }
3392
3393             /* If the new action is must-not-throw, then no region notes
3394                are created.  */
3395             if (this_action >= -1)
3396               {
3397                 call_site = add_call_site (this_landing_pad,
3398                                            this_action < 0 ? 0 : this_action);
3399                 note = emit_note_before (NOTE_INSN_EH_REGION_BEG, iter);
3400                 NOTE_EH_HANDLER (note) = call_site;
3401               }
3402
3403             last_action = this_action;
3404             last_landing_pad = this_landing_pad;
3405           }
3406         last_action_insn = iter;
3407       }
3408
3409   if (last_action >= -1 && ! first_no_action_insn)
3410     {
3411       note = emit_note_after (NOTE_INSN_EH_REGION_END, last_action_insn);
3412       NOTE_EH_HANDLER (note) = call_site;
3413     }
3414
3415   htab_delete (ar_hash);
3416 }
3417
3418 \f
3419 static void
3420 push_uleb128 (varray_type *data_area, unsigned int value)
3421 {
3422   do
3423     {
3424       unsigned char byte = value & 0x7f;
3425       value >>= 7;
3426       if (value)
3427         byte |= 0x80;
3428       VARRAY_PUSH_UCHAR (*data_area, byte);
3429     }
3430   while (value);
3431 }
3432
3433 static void
3434 push_sleb128 (varray_type *data_area, int value)
3435 {
3436   unsigned char byte;
3437   int more;
3438
3439   do
3440     {
3441       byte = value & 0x7f;
3442       value >>= 7;
3443       more = ! ((value == 0 && (byte & 0x40) == 0)
3444                 || (value == -1 && (byte & 0x40) != 0));
3445       if (more)
3446         byte |= 0x80;
3447       VARRAY_PUSH_UCHAR (*data_area, byte);
3448     }
3449   while (more);
3450 }
3451
3452 \f
3453 #ifndef HAVE_AS_LEB128
3454 static int
3455 dw2_size_of_call_site_table (void)
3456 {
3457   int n = cfun->eh->call_site_data_used;
3458   int size = n * (4 + 4 + 4);
3459   int i;
3460
3461   for (i = 0; i < n; ++i)
3462     {
3463       struct call_site_record *cs = &cfun->eh->call_site_data[i];
3464       size += size_of_uleb128 (cs->action);
3465     }
3466
3467   return size;
3468 }
3469
3470 static int
3471 sjlj_size_of_call_site_table (void)
3472 {
3473   int n = cfun->eh->call_site_data_used;
3474   int size = 0;
3475   int i;
3476
3477   for (i = 0; i < n; ++i)
3478     {
3479       struct call_site_record *cs = &cfun->eh->call_site_data[i];
3480       size += size_of_uleb128 (INTVAL (cs->landing_pad));
3481       size += size_of_uleb128 (cs->action);
3482     }
3483
3484   return size;
3485 }
3486 #endif
3487
3488 static void
3489 dw2_output_call_site_table (void)
3490 {
3491   const char *const function_start_lab
3492     = IDENTIFIER_POINTER (current_function_func_begin_label);
3493   int n = cfun->eh->call_site_data_used;
3494   int i;
3495
3496   for (i = 0; i < n; ++i)
3497     {
3498       struct call_site_record *cs = &cfun->eh->call_site_data[i];
3499       char reg_start_lab[32];
3500       char reg_end_lab[32];
3501       char landing_pad_lab[32];
3502
3503       ASM_GENERATE_INTERNAL_LABEL (reg_start_lab, "LEHB", call_site_base + i);
3504       ASM_GENERATE_INTERNAL_LABEL (reg_end_lab, "LEHE", call_site_base + i);
3505
3506       if (cs->landing_pad)
3507         ASM_GENERATE_INTERNAL_LABEL (landing_pad_lab, "L",
3508                                      CODE_LABEL_NUMBER (cs->landing_pad));
3509
3510       /* ??? Perhaps use insn length scaling if the assembler supports
3511          generic arithmetic.  */
3512       /* ??? Perhaps use attr_length to choose data1 or data2 instead of
3513          data4 if the function is small enough.  */
3514 #ifdef HAVE_AS_LEB128
3515       dw2_asm_output_delta_uleb128 (reg_start_lab, function_start_lab,
3516                                     "region %d start", i);
3517       dw2_asm_output_delta_uleb128 (reg_end_lab, reg_start_lab,
3518                                     "length");
3519       if (cs->landing_pad)
3520         dw2_asm_output_delta_uleb128 (landing_pad_lab, function_start_lab,
3521                                       "landing pad");
3522       else
3523         dw2_asm_output_data_uleb128 (0, "landing pad");
3524 #else
3525       dw2_asm_output_delta (4, reg_start_lab, function_start_lab,
3526                             "region %d start", i);
3527       dw2_asm_output_delta (4, reg_end_lab, reg_start_lab, "length");
3528       if (cs->landing_pad)
3529         dw2_asm_output_delta (4, landing_pad_lab, function_start_lab,
3530                               "landing pad");
3531       else
3532         dw2_asm_output_data (4, 0, "landing pad");
3533 #endif
3534       dw2_asm_output_data_uleb128 (cs->action, "action");
3535     }
3536
3537   call_site_base += n;
3538 }
3539
3540 static void
3541 sjlj_output_call_site_table (void)
3542 {
3543   int n = cfun->eh->call_site_data_used;
3544   int i;
3545
3546   for (i = 0; i < n; ++i)
3547     {
3548       struct call_site_record *cs = &cfun->eh->call_site_data[i];
3549
3550       dw2_asm_output_data_uleb128 (INTVAL (cs->landing_pad),
3551                                    "region %d landing pad", i);
3552       dw2_asm_output_data_uleb128 (cs->action, "action");
3553     }
3554
3555   call_site_base += n;
3556 }
3557
3558 /* Tell assembler to switch to the section for the exception handling
3559    table.  */
3560
3561 void
3562 default_exception_section (void)
3563 {
3564   if (targetm.have_named_sections)
3565     {
3566       int flags;
3567 #ifdef HAVE_LD_RO_RW_SECTION_MIXING
3568       int tt_format = ASM_PREFERRED_EH_DATA_FORMAT (/*code=*/0, /*global=*/1);
3569
3570       flags = (! flag_pic
3571                || ((tt_format & 0x70) != DW_EH_PE_absptr
3572                    && (tt_format & 0x70) != DW_EH_PE_aligned))
3573               ? 0 : SECTION_WRITE;
3574 #else
3575       flags = SECTION_WRITE;
3576 #endif
3577       named_section_flags (".gcc_except_table", flags);
3578     }
3579   else if (flag_pic)
3580     data_section ();
3581   else
3582     readonly_data_section ();
3583 }
3584
3585 void
3586 output_function_exception_table (void)
3587 {
3588   int tt_format, cs_format, lp_format, i, n;
3589 #ifdef HAVE_AS_LEB128
3590   char ttype_label[32];
3591   char cs_after_size_label[32];
3592   char cs_end_label[32];
3593 #else
3594   int call_site_len;
3595 #endif
3596   int have_tt_data;
3597   int tt_format_size = 0;
3598
3599   /* Not all functions need anything.  */
3600   if (! cfun->uses_eh_lsda)
3601     return;
3602
3603 #ifdef TARGET_UNWIND_INFO
3604   /* TODO: Move this into target file.  */
3605   fputs ("\t.personality\t", asm_out_file);
3606   output_addr_const (asm_out_file, eh_personality_libfunc);
3607   fputs ("\n\t.handlerdata\n", asm_out_file);
3608   /* Note that varasm still thinks we're in the function's code section.
3609      The ".endp" directive that will immediately follow will take us back.  */
3610 #else
3611   targetm.asm_out.exception_section ();
3612 #endif
3613
3614   have_tt_data = (VARRAY_ACTIVE_SIZE (cfun->eh->ttype_data) > 0
3615                   || VARRAY_ACTIVE_SIZE (cfun->eh->ehspec_data) > 0);
3616
3617   /* Indicate the format of the @TType entries.  */
3618   if (! have_tt_data)
3619     tt_format = DW_EH_PE_omit;
3620   else
3621     {
3622       tt_format = ASM_PREFERRED_EH_DATA_FORMAT (/*code=*/0, /*global=*/1);
3623 #ifdef HAVE_AS_LEB128
3624       ASM_GENERATE_INTERNAL_LABEL (ttype_label, "LLSDATT",
3625                                    current_function_funcdef_no);
3626 #endif
3627       tt_format_size = size_of_encoded_value (tt_format);
3628
3629       assemble_align (tt_format_size * BITS_PER_UNIT);
3630     }
3631
3632   targetm.asm_out.internal_label (asm_out_file, "LLSDA",
3633                              current_function_funcdef_no);
3634
3635   /* The LSDA header.  */
3636
3637   /* Indicate the format of the landing pad start pointer.  An omitted
3638      field implies @LPStart == @Start.  */
3639   /* Currently we always put @LPStart == @Start.  This field would
3640      be most useful in moving the landing pads completely out of
3641      line to another section, but it could also be used to minimize
3642      the size of uleb128 landing pad offsets.  */
3643   lp_format = DW_EH_PE_omit;
3644   dw2_asm_output_data (1, lp_format, "@LPStart format (%s)",
3645                        eh_data_format_name (lp_format));
3646
3647   /* @LPStart pointer would go here.  */
3648
3649   dw2_asm_output_data (1, tt_format, "@TType format (%s)",
3650                        eh_data_format_name (tt_format));
3651
3652 #ifndef HAVE_AS_LEB128
3653   if (USING_SJLJ_EXCEPTIONS)
3654     call_site_len = sjlj_size_of_call_site_table ();
3655   else
3656     call_site_len = dw2_size_of_call_site_table ();
3657 #endif
3658
3659   /* A pc-relative 4-byte displacement to the @TType data.  */
3660   if (have_tt_data)
3661     {
3662 #ifdef HAVE_AS_LEB128
3663       char ttype_after_disp_label[32];
3664       ASM_GENERATE_INTERNAL_LABEL (ttype_after_disp_label, "LLSDATTD",
3665                                    current_function_funcdef_no);
3666       dw2_asm_output_delta_uleb128 (ttype_label, ttype_after_disp_label,
3667                                     "@TType base offset");
3668       ASM_OUTPUT_LABEL (asm_out_file, ttype_after_disp_label);
3669 #else
3670       /* Ug.  Alignment queers things.  */
3671       unsigned int before_disp, after_disp, last_disp, disp;
3672
3673       before_disp = 1 + 1;
3674       after_disp = (1 + size_of_uleb128 (call_site_len)
3675                     + call_site_len
3676                     + VARRAY_ACTIVE_SIZE (cfun->eh->action_record_data)
3677                     + (VARRAY_ACTIVE_SIZE (cfun->eh->ttype_data)
3678                        * tt_format_size));
3679
3680       disp = after_disp;
3681       do
3682         {
3683           unsigned int disp_size, pad;
3684
3685           last_disp = disp;
3686           disp_size = size_of_uleb128 (disp);
3687           pad = before_disp + disp_size + after_disp;
3688           if (pad % tt_format_size)
3689             pad = tt_format_size - (pad % tt_format_size);
3690           else
3691             pad = 0;
3692           disp = after_disp + pad;
3693         }
3694       while (disp != last_disp);
3695
3696       dw2_asm_output_data_uleb128 (disp, "@TType base offset");
3697 #endif
3698     }
3699
3700   /* Indicate the format of the call-site offsets.  */
3701 #ifdef HAVE_AS_LEB128
3702   cs_format = DW_EH_PE_uleb128;
3703 #else
3704   cs_format = DW_EH_PE_udata4;
3705 #endif
3706   dw2_asm_output_data (1, cs_format, "call-site format (%s)",
3707                        eh_data_format_name (cs_format));
3708
3709 #ifdef HAVE_AS_LEB128
3710   ASM_GENERATE_INTERNAL_LABEL (cs_after_size_label, "LLSDACSB",
3711                                current_function_funcdef_no);
3712   ASM_GENERATE_INTERNAL_LABEL (cs_end_label, "LLSDACSE",
3713                                current_function_funcdef_no);
3714   dw2_asm_output_delta_uleb128 (cs_end_label, cs_after_size_label,
3715                                 "Call-site table length");
3716   ASM_OUTPUT_LABEL (asm_out_file, cs_after_size_label);
3717   if (USING_SJLJ_EXCEPTIONS)
3718     sjlj_output_call_site_table ();
3719   else
3720     dw2_output_call_site_table ();
3721   ASM_OUTPUT_LABEL (asm_out_file, cs_end_label);
3722 #else
3723   dw2_asm_output_data_uleb128 (call_site_len,"Call-site table length");
3724   if (USING_SJLJ_EXCEPTIONS)
3725     sjlj_output_call_site_table ();
3726   else
3727     dw2_output_call_site_table ();
3728 #endif
3729
3730   /* ??? Decode and interpret the data for flag_debug_asm.  */
3731   n = VARRAY_ACTIVE_SIZE (cfun->eh->action_record_data);
3732   for (i = 0; i < n; ++i)
3733     dw2_asm_output_data (1, VARRAY_UCHAR (cfun->eh->action_record_data, i),
3734                          (i ? NULL : "Action record table"));
3735
3736   if (have_tt_data)
3737     assemble_align (tt_format_size * BITS_PER_UNIT);
3738
3739   i = VARRAY_ACTIVE_SIZE (cfun->eh->ttype_data);
3740   while (i-- > 0)
3741     {
3742       tree type = VARRAY_TREE (cfun->eh->ttype_data, i);
3743       rtx value;
3744
3745       if (type == NULL_TREE)
3746         value = const0_rtx;
3747       else
3748         {
3749           struct cgraph_varpool_node *node;
3750
3751           type = lookup_type_for_runtime (type);
3752           value = expand_expr (type, NULL_RTX, VOIDmode, EXPAND_INITIALIZER);
3753
3754           /* Let cgraph know that the rtti decl is used.  Not all of the
3755              paths below go through assemble_integer, which would take
3756              care of this for us.  */
3757           STRIP_NOPS (type);
3758           if (TREE_CODE (type) == ADDR_EXPR)
3759             {
3760               type = TREE_OPERAND (type, 0);
3761               if (TREE_CODE (type) == VAR_DECL)
3762                 {
3763                   node = cgraph_varpool_node (type);
3764                   if (node)
3765                     cgraph_varpool_mark_needed_node (node);
3766                 }
3767             }
3768           else if (TREE_CODE (type) != INTEGER_CST)
3769             abort ();
3770         }
3771
3772       if (tt_format == DW_EH_PE_absptr || tt_format == DW_EH_PE_aligned)
3773         assemble_integer (value, tt_format_size,
3774                           tt_format_size * BITS_PER_UNIT, 1);
3775       else
3776         dw2_asm_output_encoded_addr_rtx (tt_format, value, NULL);
3777     }
3778
3779 #ifdef HAVE_AS_LEB128
3780   if (have_tt_data)
3781       ASM_OUTPUT_LABEL (asm_out_file, ttype_label);
3782 #endif
3783
3784   /* ??? Decode and interpret the data for flag_debug_asm.  */
3785   n = VARRAY_ACTIVE_SIZE (cfun->eh->ehspec_data);
3786   for (i = 0; i < n; ++i)
3787     dw2_asm_output_data (1, VARRAY_UCHAR (cfun->eh->ehspec_data, i),
3788                          (i ? NULL : "Exception specification table"));
3789
3790   function_section (current_function_decl);
3791 }
3792
3793 #include "gt-except.h"