OSDN Git Service

Oops.
[pf3gnuchains/gcc-fork.git] / gcc / except.c
1 /* Implements exceptiom handling.
2    Copyright (C) 1989, 92-95, 1996 Free Software Foundation, Inc.
3    Contributed by Mike Stump <mrs@cygnus.com>.
4
5 This file is part of GNU CC.
6
7 GNU CC is free software; you can redistribute it and/or modify
8 it under the terms of the GNU General Public License as published by
9 the Free Software Foundation; either version 2, or (at your option)
10 any later version.
11
12 GNU CC is distributed in the hope that it will be useful,
13 but WITHOUT ANY WARRANTY; without even the implied warranty of
14 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
15 GNU General Public License for more details.
16
17 You should have received a copy of the GNU General Public License
18 along with GNU CC; see the file COPYING.  If not, write to
19 the Free Software Foundation, 59 Temple Place - Suite 330,
20 Boston, MA 02111-1307, USA.  */
21
22
23 /* This file contains the exception handling code for the compiler.
24
25    Exception handling is a mechanism by which complex flows of control
26    can be designated.  The central concepts are the `exception region',
27    the associated `exception handler' for that region and the concept of
28    throwing an exception, and the context of the throw.
29
30    Restrictions are, the regions must be non-overlapping, they can be
31    nested, and there can be zero or more per function.  For each
32    region, there is one associated handler.  Regions only ever
33    surround possible context points of throws.  Regions with no such
34    context points can be optimized away, as they are trivial, and it
35    is not possible for the associated handler to ever be used during a
36    throw.
37
38    Semantics are, when an exception is thrown, control is transferred
39    to a handler, and the code of the exception handler is executed.
40    As control is transferred, the machine state (stack pointer, all
41    callee saved registers and possibly the frame pointer) is restored.
42
43    The handler that is selected by a throw, is the handler associated
44    with the smallest (most nested) region that contains the context of
45    the throw, if such a region exists.  If no region exists, the
46    search for a handler continues in the function that called the
47    function that contains the current context of the throw, with the
48    context of the throw then becoming that point in the code that
49    contains the call instruction.
50
51
52    One can add to the basic model the concepts of thrown exception
53    type, and thrown exception value.  Semantics are as above, except a
54    further check is done when finding a handler for the thrown
55    exception to see if the given handler can handle the thrown
56    exception based upon the exception object's type and possibly its
57    value.  A common optimization is when two regions are identical,
58    the handlers are combined into just one handler so the first check
59    of the resulting handler is for the inner (nested) region's
60    handler, and the second one is for the outer region's handler.  To
61    separate these two notions of handlers, we can call the subhandlers
62    `catch blocks', and use the name `handler' to refer to the
63    combination of the two.  Currently, this layer of functionality is
64    managed by the various front ends.
65
66
67    To mark the start of a exception handling region,
68    expand_eh_region_start () is called.  To mark the end, and
69    associate a handler for the region expand_eh_region_end () is used.
70    The front end can use this interface, if useful.  The back end
71    creates exception regions with these routines.  Another interface
72    the front end can use, is TARGET_EXPR.  TARGET_EXPR gives an
73    unwind-protect style interface a la emacs.
74
75
76    In this implementation, regions do not span more than one function.
77
78    In order to help with the task of finding the associated handler for
79    a region, an exception table is built which associates handlers
80    with regions.  A 3-tuple, containing a reference to the start, the
81    end and the handler is sufficient for the exception table.
82
83    In order to help with the task of restoring callee saved registers
84    and performing other associated function exit actions, function
85    `unwinders' can be generated within those function for which a
86    generic function unwinder called __unwind_function () cannot work.
87    Whether the generic __unwind_function can work is machine dependent
88    and possibly function dependent.  The macro DOESNT_NEEED_UNWINDER
89    decides if the current function being compiled needs an unwinder or
90    not.
91
92    The default is for unwinders to be used, as the default generic
93    function unwinder only calls abort ().  The compiler-generated per
94    function function unwinders simply modify the context of thrown
95    exception to be that of the call site, and then arrange for control
96    to be transferred to __throw instead of the function's caller on
97    return, and then return.  */
98
99
100 #include "config.h"
101 #include <stdio.h>
102 #include "rtl.h"
103 #include "tree.h"
104 #include "flags.h"
105 #include "except.h"
106 #include "function.h"
107 #include "insn-flags.h"
108 #include "expr.h"
109 #include "insn-codes.h"
110 #include "regs.h"
111 #include "hard-reg-set.h"
112 #include "insn-config.h"
113 #include "recog.h"
114 #include "output.h"
115
116 /* List of labels use for exception handlers.  Created by
117    find_exception_handler_labels for the optimization passes.  */
118
119 rtx exception_handler_labels;
120
121 /* Nonzero means that throw was used.  Used for now, because __throw
122    is emitted statically in each file.  */
123
124 int throw_used;
125
126 /* A stack used for keeping track of the currectly active exception
127    handling region.  As exceptions regions are started, an entry
128    describing the region is pushed onto this stack.  The current
129    region can be found by looking at the top of the stack, and as we
130    end regions, entries are poped.  */
131
132 struct eh_stack ehstack;
133
134 /* A queue used for tracking which exception regions have closed, but
135    whose handlers have not yet been expanded.  As we end regions, we
136    enqueue the entry onto this queue.  Entries are dequeue from the
137    queue during expand_leftover_cleanups and expand_start_all_catch,
138    and the handlers for regions are expanded in groups in an effort to
139    group all the handlers together in the same region of program space
140    to improve page performance.  We should redo things, so that we
141    either take RTL for the handler, or we expand the handler expressed
142    as a tree immediately at region end time.  */
143
144 struct eh_queue ehqueue;
145
146 /* Insns for the catch clauses.  */
147
148 rtx catch_clauses;
149
150 /* A list of actions for handlers for regions that are not yet
151    closed.  */
152
153 tree protect_list;
154
155 /* Stacks to keep track of various labels.  */
156
157 /* Keeps track of the label to resume to, should one want to resume
158    the normal control flow out of a handler.  Also used to rethrow
159    exceptions caught in handlers, as if they were physically emitted
160    inline.  */
161
162 struct label_node *caught_return_label_stack = NULL;
163
164 /* A spare data area for the front end's own use.  */
165
166 struct label_node *false_label_stack = NULL;
167
168 /* The rtx for the saved PC value.  */
169
170 rtx eh_saved_pc_rtx;
171
172 rtx expand_builtin_return_addr  PROTO((enum built_in_function, int, rtx));
173 \f
174 /* Various support routines to manipulate the various data structures
175    used by the exception handling code.  */
176
177 /* Push a label entry onto the given STACK.  */
178
179 void
180 push_label_entry (stack, rlabel, tlabel)
181      struct label_node **stack;
182      rtx rlabel;
183      tree tlabel;
184 {
185   struct label_node *newnode
186     = (struct label_node *) xmalloc (sizeof (struct label_node));
187
188   if (rlabel)
189     newnode->u.rlabel = rlabel;
190   else
191     newnode->u.tlabel = tlabel;
192   newnode->chain = *stack;
193   *stack = newnode;
194 }
195
196 /* Pop a label entry from the given STACK.  */
197
198 rtx
199 pop_label_entry (stack)
200      struct label_node **stack;
201 {
202   rtx label;
203   struct label_node *tempnode;
204
205   if (! *stack)
206     return NULL_RTX;
207
208   tempnode = *stack;
209   label = tempnode->u.rlabel;
210   *stack = (*stack)->chain;
211   free (tempnode);
212
213   return label;
214 }
215
216 /* Return the top element of the given STACK.  */
217
218 tree
219 top_label_entry (stack)
220      struct label_node **stack;
221 {
222   if (! *stack)
223     return NULL_TREE;
224
225   return (*stack)->u.tlabel;
226 }
227
228 /* Copy an entry.  */
229
230 static struct eh_entry *
231 copy_eh_entry (entry)
232      struct eh_entry *entry;
233 {
234   struct eh_entry *newentry;
235
236   newentry = (struct eh_entry *) xmalloc (sizeof (struct eh_entry));
237   bcopy ((char *) entry, (char *) newentry, sizeof (struct eh_entry));
238
239   return newentry;
240 }
241
242 /* Push an entry onto the given STACK.  */
243
244 static rtx
245 push_eh_entry (stack)
246      struct eh_stack *stack;
247 {
248   struct eh_node *node = (struct eh_node *) xmalloc (sizeof (struct eh_node));
249   struct eh_entry *entry = (struct eh_entry *) xmalloc (sizeof (struct eh_entry));
250
251   entry->start_label = gen_label_rtx ();
252   entry->end_label = gen_label_rtx ();
253   entry->exception_handler_label = gen_label_rtx ();
254   entry->finalization = NULL_TREE;
255
256   node->entry = entry;
257   node->chain = stack->top;
258   stack->top = node;
259
260   return entry->start_label;
261 }
262
263 /* Pop an entry from the given STACK.  */
264
265 static struct eh_entry *
266 pop_eh_entry (stack)
267      struct eh_stack *stack;
268 {
269   struct eh_node *tempnode;
270   struct eh_entry *tempentry;
271   
272   tempnode = stack->top;
273   tempentry = tempnode->entry;
274   stack->top = stack->top->chain;
275   free (tempnode);
276
277   return tempentry;
278 }
279
280 /* Enqueue an ENTRY onto the given QUEUE.  */
281
282 static void
283 enqueue_eh_entry (queue, entry)
284      struct eh_queue *queue;
285      struct eh_entry *entry;
286 {
287   struct eh_node *node = (struct eh_node *) xmalloc (sizeof (struct eh_node));
288
289   node->entry = entry;
290   node->chain = NULL;
291
292   if (queue->head == NULL)
293     {
294       queue->head = node;
295     }
296   else
297     {
298       queue->tail->chain = node;
299     }
300   queue->tail = node;
301 }
302
303 /* Dequeue an entry from the given QUEUE.  */
304
305 static struct eh_entry *
306 dequeue_eh_entry (queue)
307      struct eh_queue *queue;
308 {
309   struct eh_node *tempnode;
310   struct eh_entry *tempentry;
311
312   if (queue->head == NULL)
313     return NULL;
314
315   tempnode = queue->head;
316   queue->head = queue->head->chain;
317
318   tempentry = tempnode->entry;
319   free (tempnode);
320
321   return tempentry;
322 }
323 \f
324 /* Routine to see if exception exception handling is turned on.
325    DO_WARN is non-zero if we want to inform the user that exception
326    handling is turned off.  */
327
328 int
329 doing_eh (do_warn)
330      int do_warn;
331 {
332   if (! flag_exceptions)
333     {
334       static int warned = 0;
335       if (! warned && do_warn)
336         {
337           error ("exception handling disabled, use -fexceptions to enable");
338           warned = 1;
339         }
340       return 0;
341     }
342   return 1;
343 }
344
345 /* Given the return address in ADDR, compute the new pc to throw.
346    This has to work for the current frame of the current function, and
347    the one above it in the case of throw.  */
348
349 rtx
350 eh_outer_context (addr)
351      rtx addr;
352 {
353   /* First mask out any unwanted bits.  */
354 #ifdef MASK_RETURN_ADDR
355   emit_insn (gen_rtx (SET, Pmode,
356                       addr,
357                       gen_rtx (AND, Pmode,
358                                addr, MASK_RETURN_ADDR)));
359 #endif
360
361   /* Then subtract out enough to get into the prior region.  If this
362      is defined, assume we don't need to subtract anything, as it is
363      already within the region.  */
364 #if ! defined (RETURN_ADDR_OFFSET)
365   addr = plus_constant (addr, -1);
366 #endif
367
368   return addr;
369 }
370
371 /* Output a note marking the start of an exception handling region.  */
372
373 void
374 expand_eh_region_start ()
375 {
376   rtx note;
377
378   /* This is the old code.  */
379   if (! doing_eh (0))
380     return;
381
382 #if 0
383   /* Maybe do this to prevent jumping in and so on...  */
384   pushlevel (0);
385 #endif
386
387   note = emit_note (NULL_PTR, NOTE_INSN_EH_REGION_BEG);
388   emit_label (push_eh_entry (&ehstack));
389   NOTE_BLOCK_NUMBER (note)
390     = CODE_LABEL_NUMBER (ehstack.top->entry->exception_handler_label);
391 }
392
393 /* Output a note marking the end of an exception handling region.
394    HANDLER is the the handler for the exception region.  */
395
396 void
397 expand_eh_region_end (handler)
398      tree handler;
399 {
400   rtx note;
401
402   struct eh_entry *entry;
403
404   if (! doing_eh (0))
405     return;
406
407   entry = pop_eh_entry (&ehstack);
408
409   note = emit_note (NULL_PTR, NOTE_INSN_EH_REGION_END);
410   NOTE_BLOCK_NUMBER (note) = CODE_LABEL_NUMBER (entry->exception_handler_label);
411
412   emit_label (entry->end_label);
413
414   /* Put in something that takes up space, as otherwise the end
415      address for the EH region could have the exact same address as
416      the outer region, causing us to miss the fact that resuming
417      exception handling with this PC value would be inside the outer
418      region.  */
419   emit_insn (gen_nop ());
420
421   entry->finalization = handler;
422
423   enqueue_eh_entry (&ehqueue, entry);
424
425
426 #if 0
427   /* Makebe do this to prevent jumping in and so on...  */
428   poplevel (1, 0, 0);
429 #endif
430 }
431
432 /* Emit a call to __throw and note that we threw something.  */
433
434 static void
435 emit_throw ()
436 {
437 #ifdef JUMP_TO_THROW
438   emit_indirect_jump (throw_libfunc);
439 #else
440   SYMBOL_REF_USED (throw_libfunc) = 1;
441   emit_library_call (throw_libfunc, 0, VOIDmode, 0);
442 #endif
443   throw_used = 1;
444   emit_barrier ();
445 }
446
447 /* An internal throw with an indirect CONTEXT we want to throw from.  */
448
449 void
450 expand_internal_throw_indirect (context)
451      rtx context;
452 {
453   emit_move_insn (eh_saved_pc_rtx, context);
454   emit_throw ();
455 }
456
457 /* An internal throw with a direct CONTEXT we want to throw from.  The
458    context should be a label.  */
459
460 void
461 expand_internal_throw (context)
462      rtx context;
463 {
464   expand_internal_throw_indirect (gen_rtx (LABEL_REF, Pmode, context));
465 }
466
467 /* Called from expand_exception_blocks and expand_end_catch_block to
468    expand any pending handlers.  */
469
470 void
471 expand_leftover_cleanups ()
472 {
473   struct eh_entry *entry;
474
475   while ((entry = dequeue_eh_entry (&ehqueue)) != 0)
476     {
477       rtx prev;
478
479       emit_label (entry->exception_handler_label);
480
481       expand_expr (entry->finalization, const0_rtx, VOIDmode, 0);
482
483       prev = get_last_insn ();
484       if (! (prev && GET_CODE (prev) == BARRIER))
485         {
486           /* The below can be optimized away, and we could just fall into the
487              next EH handler, if we are certain they are nested.  */
488           /* Code to throw out to outer context, if we fall off end of the
489              handler.  */
490           expand_internal_throw (entry->end_label);
491         }
492
493       /* leftover try block, opps.  */
494       if (entry->finalization == integer_zero_node)
495         abort ();
496
497       free (entry);
498     }
499 }
500
501 /* Generate RTL for the start of all the catch blocks.  Used for
502    arranging for the exception handling code to be placed farther out
503    of line than normal.  */
504
505 void
506 expand_start_all_catch ()
507 {
508   struct eh_entry *entry;
509   tree label;
510
511   if (! doing_eh (1))
512     return;
513
514   emit_line_note (input_filename, lineno);
515   label = build_decl (LABEL_DECL, NULL_TREE, NULL_TREE);
516
517   /* The label for the exception handling block we will save.  This is
518      Lresume, in the documention.  */
519   expand_label (label);
520   
521   /* Put in something that takes up space, as otherwise the end
522      address for the EH region could have the exact same address as
523      the outer region, causing us to miss the fact that resuming
524      exception handling with this PC value would be inside the outer
525      region.  */
526   emit_insn (gen_nop ());
527
528   push_label_entry (&caught_return_label_stack, NULL_RTX, label);
529
530   /* Start a new sequence for all the catch blocks.  We will add this
531      to the gloabl sequence catch_clauses, when we have completed all
532      the handlers in this handler-seq.  */
533   start_sequence ();
534
535   while (1)
536     {
537       rtx prev;
538
539       entry = dequeue_eh_entry (&ehqueue);
540       emit_label (entry->exception_handler_label);
541
542       expand_expr (entry->finalization, const0_rtx, VOIDmode, 0);
543
544       /* When we get down to the matching entry, stop.  */
545       if (entry->finalization == integer_zero_node)
546         break;
547
548       prev = get_last_insn ();
549       if (! (prev && GET_CODE (prev) == BARRIER))
550         {
551           /* The below can be optimized away, and we could just fall into the
552              next EH handler, if we are certain they are nested.  */
553           /* Code to throw out to outer context, if we fall off end of the
554              handler.  */
555           expand_internal_throw (entry->end_label);
556         }
557
558       free (entry);
559     }
560 }
561
562 /* Generate RTL for the end of all the catch blocks.  */
563
564 void
565 expand_end_all_catch ()
566 {
567   rtx new_catch_clause;
568
569   if (! doing_eh (1))
570     return;
571
572   /* Code to throw out to outer context, if we fall off end of catch
573      handlers.  This is rethrow (Lresume, same id, same obj); in the
574      documentation.  */
575   expand_internal_throw (DECL_RTL (top_label_entry (&caught_return_label_stack)));
576
577   /* Now we have the complete catch sequence.  */
578   new_catch_clause = get_insns ();
579   end_sequence ();
580   
581   /* This level of catch blocks is done, so set up the successful
582      catch jump label for the next layer of catch blocks.  */
583   pop_label_entry (&caught_return_label_stack);
584
585   /* Add the new sequence of catches to the main one for this function.  */
586   push_to_sequence (catch_clauses);
587   emit_insns (new_catch_clause);
588   catch_clauses = get_insns ();
589   end_sequence ();
590   
591   /* Here we fall through into the continuation code.  */
592 }
593
594 /* End all the pending exception regions from protect_list that have
595    been started, but not yet completed.  */
596
597 void
598 end_protect_partials ()
599 {
600   while (protect_list)
601     {
602       expand_eh_region_end (TREE_VALUE (protect_list));
603       protect_list = TREE_CHAIN (protect_list);
604     }
605 }
606 \f
607 /* The exception table that we build that is used for looking up and
608    dispatching exceptions, it's size, and it's maximum size before we
609    have to extend it.  */
610 static int *eh_table;
611 static int eh_table_size;
612 static int eh_table_max_size;
613
614 /* Note the need for an exception table entry for region N.  If we
615    don't need to output an explicit exception table, avoid all the
616    extra work.  Called during final_scan_insn time.  */
617
618 void
619 add_eh_table_entry (n)
620      int n;
621 {
622 #ifndef OMIT_EH_TABLE
623   if (eh_table_size >= eh_table_max_size)
624     {
625       if (eh_table)
626         {
627           eh_table_max_size += eh_table_max_size>>1;
628
629           if (eh_table_max_size < 0)
630             abort ();
631
632           if ((eh_table = (int *) realloc (eh_table, eh_table_max_size))
633               == 0)
634             fatal ("virtual memory exhausted");
635         }
636       else
637         {
638           eh_table_max_size = 252;
639           eh_table = (int *) xmalloc (eh_table_max_size * sizeof (int));
640         }
641     }
642   eh_table[eh_table_size++] = n;
643 #endif
644 }
645
646 /* Conditional to test to see if we need to output an exception table.
647    Note, on some platforms, we don't have to output a table
648    explicitly.  This routine doesn't mean we don't have one.  */
649
650 int
651 exception_table_p ()
652 {
653   if (eh_table)
654     return 1;
655
656   return 0;
657 }
658
659 /* Output an entry N for the exception table to the specified FILE.  */
660
661 static void
662 output_exception_table_entry (file, n)
663      FILE *file;
664      int n;
665 {
666   char buf[256];
667   rtx sym;
668
669   ASM_GENERATE_INTERNAL_LABEL (buf, "LEHB", n);
670   sym = gen_rtx (SYMBOL_REF, Pmode, buf);
671   assemble_integer (sym, POINTER_SIZE / BITS_PER_UNIT, 1);
672
673   ASM_GENERATE_INTERNAL_LABEL (buf, "LEHE", n);
674   sym = gen_rtx (SYMBOL_REF, Pmode, buf);
675   assemble_integer (sym, POINTER_SIZE / BITS_PER_UNIT, 1);
676
677   ASM_GENERATE_INTERNAL_LABEL (buf, "L", n);
678   sym = gen_rtx (SYMBOL_REF, Pmode, buf);
679   assemble_integer (sym, POINTER_SIZE / BITS_PER_UNIT, 1);
680
681   putc ('\n', file);            /* blank line */
682 }
683
684 /* Output the exception table if we have one and need one.  */
685
686 void
687 output_exception_table ()
688 {
689   int i;
690   extern FILE *asm_out_file;
691
692   if (! doing_eh (0))
693     return;
694
695   exception_section ();
696
697   /* Beginning marker for table.  */
698   assemble_align (GET_MODE_ALIGNMENT (ptr_mode));
699   assemble_label ("__EXCEPTION_TABLE__");
700
701   assemble_integer (const0_rtx, POINTER_SIZE / BITS_PER_UNIT, 1);
702   assemble_integer (const0_rtx, POINTER_SIZE / BITS_PER_UNIT, 1);
703   assemble_integer (const0_rtx, POINTER_SIZE / BITS_PER_UNIT, 1);
704   putc ('\n', asm_out_file);            /* blank line */
705
706   for (i = 0; i < eh_table_size; ++i)
707     output_exception_table_entry (asm_out_file, eh_table[i]);
708
709   free (eh_table);
710
711   /* Ending marker for table.  */
712   assemble_label ("__EXCEPTION_END__");
713   assemble_integer (constm1_rtx, POINTER_SIZE / BITS_PER_UNIT, 1);
714   assemble_integer (constm1_rtx, POINTER_SIZE / BITS_PER_UNIT, 1);
715   assemble_integer (constm1_rtx, POINTER_SIZE / BITS_PER_UNIT, 1);
716   putc ('\n', asm_out_file);            /* blank line */
717 }
718
719 /* Generate code to initialize the exception table at program startup
720    time.  */
721
722 void
723 register_exception_table ()
724 {
725   emit_library_call (gen_rtx (SYMBOL_REF, Pmode, "__register_exceptions"), 0,
726                      VOIDmode, 1,
727                      gen_rtx (SYMBOL_REF, Pmode, "__EXCEPTION_TABLE__"),
728                      Pmode);
729 }
730 \f
731 /* Emit the RTL for the start of the per function unwinder for the
732    current function.  */
733
734 void
735 start_eh_unwinder ()
736 {
737 #ifdef DOESNT_NEED_UNWINDER
738   if (DOESNT_NEED_UNWINDER)
739     return;
740 #endif
741
742   expand_eh_region_start ();
743 }
744
745 /* Emit the RTL for the end of the per function unwinder for the
746    current function.  */
747
748 void
749 end_eh_unwinder ()
750 {
751   tree expr;
752   rtx return_val_rtx, ret_val, label, end, insns;
753
754   if (! doing_eh (0))
755     return;
756
757 #ifdef DOESNT_NEED_UNWINDER
758   if (DOESNT_NEED_UNWINDER)
759     return;
760 #endif
761
762   expr = make_node (RTL_EXPR);
763   TREE_TYPE (expr) = void_type_node;
764   RTL_EXPR_RTL (expr) = const0_rtx;
765   TREE_SIDE_EFFECTS (expr) = 1;
766   start_sequence_for_rtl_expr (expr);
767
768   ret_val = expand_builtin_return_addr (BUILT_IN_RETURN_ADDRESS,
769                                         0, hard_frame_pointer_rtx);
770   return_val_rtx = copy_to_reg (ret_val);
771
772   return_val_rtx = eh_outer_context (return_val_rtx);
773
774   emit_move_insn (eh_saved_pc_rtx, return_val_rtx);
775   
776 #ifdef JUMP_TO_THROW
777   emit_move_insn (ret_val, throw_libfunc);
778 #else
779   label = gen_label_rtx ();
780   emit_move_insn (ret_val, gen_rtx (LABEL_REF, Pmode, label));
781 #endif
782
783 #ifdef RETURN_ADDR_OFFSET
784   return_val_rtx = plus_constant (ret_val, -RETURN_ADDR_OFFSET);
785   if (return_val_rtx != ret_val)
786     emit_move_insn (ret_val, return_val_rtx);
787 #endif
788   
789   end = gen_label_rtx ();
790   emit_jump (end);  
791
792   RTL_EXPR_SEQUENCE (expr) = get_insns ();
793   end_sequence ();
794   expand_eh_region_end (expr);
795
796   emit_jump (end);
797
798 #ifndef JUMP_TO_THROW
799   emit_label (label);
800   emit_throw ();
801 #endif
802   
803   expand_leftover_cleanups ();
804
805   emit_label (end);
806 }
807
808 /* Emit the RTL for the per function unwinder for the current
809    function, if needed.  Called after all the code that needs unwind
810    protection is output.  */
811
812 void
813 emit_unwinder ()
814 {
815   rtx insns;
816
817   start_sequence ();
818   start_eh_unwinder ();
819   insns = get_insns ();
820   end_sequence ();
821
822   if (insns)
823     emit_insns_after (insns, get_insns ());
824
825   end_eh_unwinder ();
826 }
827
828 /* Scan the current insns and build a list of handler labels.  Called
829    after the last exception handling region is added to the current
830    function (when the rtl is almost all built for the current
831    function) and before the jump optimization pass.  */
832
833 void
834 find_exception_handler_labels ()
835 {
836   rtx insn;
837   int max_labelno = max_label_num ();
838   int min_labelno = get_first_label_num ();
839   rtx *labels;
840
841   exception_handler_labels = NULL_RTX;
842
843   /* If we aren't doing exception handling, there isn't much to check.  */
844   if (! doing_eh (0))
845     return;
846
847   /* First we generate a handy reference to each label.  */
848
849   labels = (rtx *) alloca ((max_labelno - min_labelno) * sizeof (rtx));
850   labels -= min_labelno;
851
852   for (insn = get_insns (); insn; insn = NEXT_INSN (insn))
853     {
854       if (GET_CODE (insn) == CODE_LABEL)
855         if (CODE_LABEL_NUMBER (insn) >= min_labelno
856             && CODE_LABEL_NUMBER (insn) < max_labelno)
857           labels[CODE_LABEL_NUMBER (insn)] = insn;
858     }
859
860   /* Then for each start of a region, we add its label to the list.  */
861   for (insn = get_insns (); insn; insn = NEXT_INSN (insn))
862     {
863       if (GET_CODE (insn) == NOTE
864           && NOTE_LINE_NUMBER (insn) == NOTE_INSN_EH_REGION_BEG)
865         {
866           rtx label = NULL_RTX;
867
868           if (NOTE_BLOCK_NUMBER (insn) >= min_labelno
869               && NOTE_BLOCK_NUMBER (insn) < max_labelno)
870             {
871               label = labels[NOTE_BLOCK_NUMBER (insn)];
872
873               if (label)
874                 exception_handler_labels
875                   = gen_rtx (EXPR_LIST, VOIDmode,
876                              label, exception_handler_labels);
877               else
878                 warning ("didn't find handler for EH region %d",
879                          NOTE_BLOCK_NUMBER (insn));
880             }
881           else
882             warning ("mismatched EH region %d", NOTE_BLOCK_NUMBER (insn));
883         }
884     }
885 }
886
887 /* Do some sanity checking on the exception_handler_labels list.  Can
888    be called after find_exception_handler_labels is called to build
889    the list of exception handlers for the current function, and before
890    we finish processing the current function.  */
891
892 void
893 check_exception_handler_labels ()
894 {
895   rtx insn, handler;
896
897   /* If we aren't doing exception handling, there isn't much to check.  */
898   if (! doing_eh (0))
899     return;
900
901   for (handler = exception_handler_labels;
902        handler;
903        handler = XEXP (handler, 1))
904     {
905       for (insn = get_insns (); insn; insn = NEXT_INSN (insn))
906         {
907           if (GET_CODE (insn) == CODE_LABEL)
908             {
909               if (CODE_LABEL_NUMBER (insn)
910                   == CODE_LABEL_NUMBER (XEXP (handler, 0)))
911                 {
912                   if (insn != XEXP (handler, 0))
913                     warning ("mismatched handler %d",
914                              CODE_LABEL_NUMBER (insn));
915                   break;
916                 }
917             }
918         }
919       if (insn == NULL_RTX)
920         warning ("handler not found %d",
921                  CODE_LABEL_NUMBER (XEXP (handler, 0)));
922     }
923
924   /* Now go through, and make sure that for each region we have, that we
925      have the corresponding label.  */
926   for (insn = get_insns (); insn; insn = NEXT_INSN (insn))
927     {
928       if (GET_CODE (insn) == NOTE
929           && (NOTE_LINE_NUMBER (insn) == NOTE_INSN_EH_REGION_BEG ||
930               NOTE_LINE_NUMBER (insn) == NOTE_INSN_EH_REGION_END))
931         {
932           for (handler = exception_handler_labels;
933                handler;
934                handler = XEXP (handler, 1))
935             {
936               if (CODE_LABEL_NUMBER (XEXP (handler, 0))
937                   == NOTE_BLOCK_NUMBER (insn))
938                 break;
939             }
940           if (handler == NULL_RTX)
941             warning ("region exists, no handler %d",
942                      NOTE_BLOCK_NUMBER (insn));
943         }
944     }
945 }
946 \f
947 /* This group of functions initializes the exception handling data
948    structures at the start of the compilation, initializes the data
949    structures at the start of a function, saves and restores the
950    exception handling data structures for the start/end of a nested
951    function.  */
952
953 /* Toplevel initialization for EH things.  */ 
954
955 void
956 init_eh ()
957 {
958   eh_saved_pc_rtx = gen_rtx (MEM, ptr_mode,
959                              gen_rtx (SYMBOL_REF, Pmode, "__eh_pc"));
960 }
961
962 /* Initialize various EH things.  */
963
964 void
965 init_eh_for_function ()
966 {
967   ehstack.top = 0;
968   ehqueue.head = ehqueue.tail = 0;
969   catch_clauses = NULL_RTX;
970   false_label_stack = 0;
971   caught_return_label_stack = 0;
972   protect_list = NULL_TREE;
973 }
974
975 /* Save various EH things for the current function into the save area
976    denoted by P.  */
977
978 void
979 save_eh_status (p)
980      struct function *p;
981 {
982   p->ehstack = ehstack;
983   p->ehqueue = ehqueue;
984   p->catch_clauses = catch_clauses;
985   p->false_label_stack = false_label_stack;
986   p->caught_return_label_stack = caught_return_label_stack;
987   p->protect_list = protect_list;
988
989   init_eh ();
990 }
991
992 /* Restore various EH things for the current function from the save
993    area denoted by P.  */
994
995 void
996 restore_eh_status (p)
997      struct function *p;
998 {
999   protect_list = p->protect_list;
1000   caught_return_label_stack = p->caught_return_label_stack;
1001   false_label_stack = p->false_label_stack;
1002   catch_clauses = p->catch_clauses;
1003   ehqueue = p->ehqueue;
1004   ehstack = p->ehstack;
1005 }
1006 \f
1007 /* This section is for the exception handling specific optimization
1008    pass.  First are the internal routines, and then the main
1009    optimization pass.  */
1010
1011 /* Determine if the given INSN can throw an exception.  */
1012
1013 static int
1014 can_throw (insn)
1015      rtx insn;
1016 {
1017   /* The only things that can possibly throw are calls.  */
1018   if (GET_CODE (insn) == CALL_INSN)
1019     return 1;
1020
1021 #ifdef ASYNCH_EXCEPTIONS
1022   /* If we wanted asynchronous exceptions, then everything but NOTEs
1023      and CODE_LABELs could throw.  */
1024   if (GET_CODE (insn) != NOTE && GET_CODE (insn) != CODE_LABEL)
1025     return 1;
1026 #endif
1027
1028   return 0;
1029 }
1030
1031 /* Scan a region, looking for a matching end, and decide if the region
1032    can be removed.  INSN is the start of the region, N is the region
1033    number, and DELETE_OUTER is to note if anything in this region can
1034    throw.  */
1035
1036 static rtx
1037 scan_region (insn, n, delete_outer)
1038      rtx insn;
1039      int n;
1040      int *delete_outer;
1041 {
1042   rtx start = insn;
1043
1044   /* Assume we can delete the region.  */
1045   int delete = 1;
1046
1047   insn = NEXT_INSN (insn);
1048
1049   /* Look for the matching end.  */
1050   while (! (GET_CODE (insn) == NOTE
1051             && NOTE_LINE_NUMBER (insn) == NOTE_INSN_EH_REGION_END))
1052     {
1053       /* If anything can throw, we can't remove the region.  */
1054       if (delete && can_throw (insn))
1055         {
1056           delete = 0;
1057         }
1058
1059       /* Watch out for and handle nested regions.  */
1060       if (GET_CODE (insn) == NOTE
1061           && NOTE_LINE_NUMBER (insn) == NOTE_INSN_EH_REGION_BEG)
1062         {
1063           insn = scan_region (insn, NOTE_BLOCK_NUMBER (insn), &delete);
1064         }
1065
1066       insn = NEXT_INSN (insn);
1067     }
1068
1069   /* The _BEG/_END NOTEs must match and nest.  */
1070   if (NOTE_BLOCK_NUMBER (insn) != n)
1071     abort ();
1072
1073   /* If anything can throw, we can throw.  */
1074   if (! delete)
1075     *delete_outer = 0;
1076   else
1077     {
1078       /* Delete the start and end of the region.  */
1079       delete_insn (start);
1080       delete_insn (insn);
1081
1082       /* Only do this part if we have built the exception handler
1083          labels.  */
1084       if (exception_handler_labels)
1085         {
1086           rtx x, *prev = &exception_handler_labels;
1087
1088           /* Find it in the list of handlers.  */
1089           for (x = exception_handler_labels; x; x = XEXP (x, 1))
1090             {
1091               rtx label = XEXP (x, 0);
1092               if (CODE_LABEL_NUMBER (label) == n)
1093                 {
1094                   /* If we are the last reference to the handler,
1095                      delete it.  */
1096                   if (--LABEL_NUSES (label) == 0)
1097                     delete_insn (label);
1098
1099                   if (optimize)
1100                     {
1101                       /* Remove it from the list of exception handler
1102                          labels, if we are optimizing.  If we are not, then
1103                          leave it in the list, as we are not really going to
1104                          remove the region.  */
1105                       *prev = XEXP (x, 1);
1106                       XEXP (x, 1) = 0;
1107                       XEXP (x, 0) = 0;
1108                     }
1109
1110                   break;
1111                 }
1112               prev = &XEXP (x, 1);
1113             }
1114         }
1115     }
1116   return insn;
1117 }
1118
1119 /* Perform various interesting optimizations for exception handling
1120    code.
1121
1122    We find empty exception regions, and remove them.  The jump
1123    optimization code will remove the handler if nothing else uses it.  */
1124
1125 void
1126 exception_optimize ()
1127 {
1128   rtx insn, regions = NULL_RTX;
1129   int n;
1130
1131   /* First remove empty regions.  */
1132   for (insn = get_insns (); insn; insn = NEXT_INSN (insn))
1133     {
1134       if (GET_CODE (insn) == NOTE
1135           && NOTE_LINE_NUMBER (insn) == NOTE_INSN_EH_REGION_BEG)
1136         {
1137           insn = scan_region (insn, NOTE_BLOCK_NUMBER (insn), &n);
1138         }
1139     }
1140 }