]> gcc.gnu.org Git - gcc.git/blob - gcc/cp/except.c
[multiple changes]
[gcc.git] / gcc / cp / except.c
1 /* Handle exceptional things in C++.
2 Copyright (C) 1989, 92-97, 1998 Free Software Foundation, Inc.
3 Contributed by Michael Tiemann <tiemann@cygnus.com>
4 Rewritten by Mike Stump <mrs@cygnus.com>, based upon an
5 initial re-implementation courtesy Tad Hunt.
6
7 This file is part of GNU CC.
8
9 GNU CC is free software; you can redistribute it and/or modify
10 it under the terms of the GNU General Public License as published by
11 the Free Software Foundation; either version 2, or (at your option)
12 any later version.
13
14 GNU CC is distributed in the hope that it will be useful,
15 but WITHOUT ANY WARRANTY; without even the implied warranty of
16 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
17 GNU General Public License for more details.
18
19 You should have received a copy of the GNU General Public License
20 along with GNU CC; see the file COPYING. If not, write to
21 the Free Software Foundation, 59 Temple Place - Suite 330,
22 Boston, MA 02111-1307, USA. */
23
24
25 #include "config.h"
26 #include "system.h"
27 #include "tree.h"
28 #include "rtl.h"
29 #include "cp-tree.h"
30 #include "flags.h"
31 #include "obstack.h"
32 #include "expr.h"
33 #include "output.h"
34 #include "except.h"
35 #include "function.h"
36 #include "defaults.h"
37 #include "toplev.h"
38 #include "eh-common.h"
39
40 rtx expand_builtin_return_addr PROTO((enum built_in_function, int, rtx));
41
42 /* Holds the fndecl for __builtin_return_address. */
43 tree builtin_return_address_fndecl;
44
45 /* A couple of backend routines from m88k.c */
46
47 static void push_eh_cleanup PROTO((void));
48 static tree build_eh_type_type PROTO((tree));
49 static tree build_eh_type PROTO((tree));
50 static void expand_end_eh_spec PROTO((tree));
51 static tree call_eh_info PROTO((void));
52 static void push_eh_info PROTO((void));
53 static tree get_eh_info PROTO((void));
54 static tree get_eh_value PROTO((void));
55 static tree get_eh_type PROTO((void));
56 static tree get_eh_caught PROTO((void));
57 static tree get_eh_handlers PROTO((void));
58 static tree do_pop_exception PROTO((void));
59 static void process_start_catch_block PROTO((tree, tree));
60 static void process_start_catch_block_old PROTO((tree, tree));
61 static tree build_eh_type_type_ref PROTO((tree));
62
63 #if 0
64 /* This is the startup, and finish stuff per exception table. */
65
66 /* XXX - Tad: exception handling section */
67 #ifndef EXCEPT_SECTION_ASM_OP
68 #define EXCEPT_SECTION_ASM_OP "section\t.gcc_except_table,\"a\",@progbits"
69 #endif
70
71 #ifdef EXCEPT_SECTION_ASM_OP
72
73 /* on machines which support it, the exception table lives in another section,
74 but it needs a label so we can reference it... This sets up that
75 label! */
76 asm (EXCEPT_SECTION_ASM_OP);
77 exception_table __EXCEPTION_TABLE__[1] = { (void*)0, (void*)0, (void*)0 };
78 asm (TEXT_SECTION_ASM_OP);
79
80 #endif /* EXCEPT_SECTION_ASM_OP */
81
82 #ifdef EXCEPT_SECTION_ASM_OP
83
84 /* we need to know where the end of the exception table is... so this
85 is how we do it! */
86
87 asm (EXCEPT_SECTION_ASM_OP);
88 exception_table __EXCEPTION_END__[1] = { (void*)-1, (void*)-1, (void*)-1 };
89 asm (TEXT_SECTION_ASM_OP);
90
91 #endif /* EXCEPT_SECTION_ASM_OP */
92
93 #endif
94
95 #include "decl.h"
96 #include "insn-flags.h"
97 #include "obstack.h"
98
99 /* ======================================================================
100 Briefly the algorithm works like this:
101
102 When a constructor or start of a try block is encountered,
103 push_eh_entry (&eh_stack) is called. Push_eh_entry () creates a
104 new entry in the unwind protection stack and returns a label to
105 output to start the protection for that block.
106
107 When a destructor or end try block is encountered, pop_eh_entry
108 (&eh_stack) is called. Pop_eh_entry () returns the eh_entry it
109 created when push_eh_entry () was called. The eh_entry structure
110 contains three things at this point. The start protect label,
111 the end protect label, and the exception handler label. The end
112 protect label should be output before the call to the destructor
113 (if any). If it was a destructor, then its parse tree is stored
114 in the finalization variable in the eh_entry structure. Otherwise
115 the finalization variable is set to NULL to reflect the fact that
116 it is the end of a try block. Next, this modified eh_entry node
117 is enqueued in the finalizations queue by calling
118 enqueue_eh_entry (&queue,entry).
119
120 +---------------------------------------------------------------+
121 |XXX: Will need modification to deal with partially |
122 | constructed arrays of objects |
123 | |
124 | Basically, this consists of keeping track of how many |
125 | of the objects have been constructed already (this |
126 | should be in a register though, so that shouldn't be a |
127 | problem. |
128 +---------------------------------------------------------------+
129
130 When a catch block is encountered, there is a lot of work to be
131 done.
132
133 Since we don't want to generate the catch block inline with the
134 regular flow of the function, we need to have some way of doing
135 so. Luckily, we can use sequences to defer the catch sections.
136 When the start of a catch block is encountered, we start the
137 sequence. After the catch block is generated, we end the
138 sequence.
139
140 Next we must insure that when the catch block is executed, all
141 finalizations for the matching try block have been completed. If
142 any of those finalizations throw an exception, we must call
143 terminate according to the ARM (section r.15.6.1). What this
144 means is that we need to dequeue and emit finalizations for each
145 entry in the eh_queue until we get to an entry with a NULL
146 finalization field. For any of the finalization entries, if it
147 is not a call to terminate (), we must protect it by giving it
148 another start label, end label, and exception handler label,
149 setting its finalization tree to be a call to terminate (), and
150 enqueue'ing this new eh_entry to be output at an outer level.
151 Finally, after all that is done, we can get around to outputting
152 the catch block which basically wraps all the "catch (...) {...}"
153 statements in a big if/then/else construct that matches the
154 correct block to call.
155
156 ===================================================================== */
157
158 /* local globals for function calls
159 ====================================================================== */
160
161 /* Used to cache "terminate" and "__throw_type_match*". */
162 static tree Terminate, CatchMatch;
163
164 /* Used to cache __find_first_exception_table_match for throw. */
165 static tree FirstExceptionMatch;
166
167 /* Used to cache a call to __unwind_function. */
168 static tree Unwind;
169
170 /* ====================================================================== */
171
172
173 /* ========================================================================= */
174
175
176
177 /* local globals - these local globals are for storing data necessary for
178 generating the exception table and code in the correct order.
179
180 ========================================================================= */
181
182 extern rtx catch_clauses;
183 extern tree const_ptr_type_node;
184
185 /* ========================================================================= */
186
187 /* sets up all the global eh stuff that needs to be initialized at the
188 start of compilation.
189
190 This includes:
191 - Setting up all the function call trees. */
192
193 void
194 init_exception_processing ()
195 {
196 /* void vtype () */
197 tree vtype = build_function_type (void_type_node, void_list_node);
198
199 if (flag_honor_std)
200 push_namespace (get_identifier ("std"));
201 Terminate = auto_function (get_identifier ("terminate"),
202 vtype, NOT_BUILT_IN);
203 TREE_THIS_VOLATILE (Terminate) = 1;
204 if (flag_honor_std)
205 pop_namespace ();
206
207 push_lang_context (lang_name_c);
208
209 set_exception_lang_code (EH_LANG_C_plus_plus);
210 set_exception_version_code (1);
211
212 CatchMatch
213 = builtin_function (flag_rtti
214 ? "__throw_type_match_rtti"
215 : "__throw_type_match",
216 build_function_type (ptr_type_node,
217 tree_cons (NULL_TREE, const_ptr_type_node,
218 tree_cons (NULL_TREE, const_ptr_type_node,
219 tree_cons (NULL_TREE, ptr_type_node,
220 void_list_node)))),
221 NOT_BUILT_IN, NULL_PTR);
222 FirstExceptionMatch
223 = builtin_function ("__find_first_exception_table_match",
224 build_function_type (ptr_type_node,
225 tree_cons (NULL_TREE, ptr_type_node,
226 void_list_node)),
227 NOT_BUILT_IN, NULL_PTR);
228 Unwind
229 = builtin_function ("__unwind_function",
230 build_function_type (void_type_node,
231 tree_cons (NULL_TREE, ptr_type_node,
232 void_list_node)),
233 NOT_BUILT_IN, NULL_PTR);
234
235 pop_lang_context ();
236
237 /* If we use setjmp/longjmp EH, arrange for all cleanup actions to
238 be protected with __terminate. */
239 protect_cleanup_actions_with_terminate = 1;
240 }
241
242 /* Retrieve a pointer to the cp_eh_info node for the current exception. */
243
244 static tree
245 call_eh_info ()
246 {
247 tree fn;
248
249 fn = get_identifier ("__cp_eh_info");
250 if (IDENTIFIER_GLOBAL_VALUE (fn))
251 fn = IDENTIFIER_GLOBAL_VALUE (fn);
252 else
253 {
254 tree t1, t, fields[7];
255
256 /* Declare cp_eh_info * __cp_eh_info (void),
257 as defined in exception.cc. */
258 push_obstacks_nochange ();
259 end_temporary_allocation ();
260
261 /* struct cp_eh_info. This must match exception.cc. Note that this
262 type is not pushed anywhere. */
263 t1= make_lang_type (RECORD_TYPE);
264 fields[0] = build_lang_field_decl (FIELD_DECL,
265 get_identifier ("handler_label"), ptr_type_node);
266 fields[1] = build_lang_field_decl (FIELD_DECL,
267 get_identifier ("dynamic_handler_chain"), ptr_type_node);
268 fields[2] = build_lang_field_decl (FIELD_DECL,
269 get_identifier ("info"), ptr_type_node);
270 /* N.B.: The fourth field LEN is expected to be
271 the number of fields - 1, not the total number of fields. */
272 finish_builtin_type (t1, "eh_context", fields, 2, ptr_type_node);
273 t1 = build_pointer_type (t1);
274
275 t1= make_lang_type (RECORD_TYPE);
276 fields[0] = build_lang_field_decl (FIELD_DECL,
277 get_identifier ("match_function"), ptr_type_node);
278 fields[1] = build_lang_field_decl (FIELD_DECL,
279 get_identifier ("language"), short_integer_type_node);
280 fields[2] = build_lang_field_decl (FIELD_DECL,
281 get_identifier ("version"), short_integer_type_node);
282 /* N.B.: The fourth field LEN is expected to be
283 the number of fields - 1, not the total number of fields. */
284 finish_builtin_type (t1, "__eh_info", fields, 2, ptr_type_node);
285 t = make_lang_type (RECORD_TYPE);
286 fields[0] = build_lang_field_decl (FIELD_DECL,
287 get_identifier ("eh_info"), t1);
288 fields[1] = build_lang_field_decl (FIELD_DECL, get_identifier ("value"),
289 ptr_type_node);
290 fields[2] = build_lang_field_decl (FIELD_DECL, get_identifier ("type"),
291 ptr_type_node);
292 fields[3] = build_lang_field_decl
293 (FIELD_DECL, get_identifier ("cleanup"),
294 build_pointer_type (build_function_type
295 (ptr_type_node, tree_cons
296 (NULL_TREE, ptr_type_node, void_list_node))));
297 fields[4] = build_lang_field_decl (FIELD_DECL, get_identifier ("caught"),
298 boolean_type_node);
299 fields[5] = build_lang_field_decl (FIELD_DECL, get_identifier ("next"),
300 build_pointer_type (t));
301 fields[6] = build_lang_field_decl
302 (FIELD_DECL, get_identifier ("handlers"), long_integer_type_node);
303 /* N.B.: The fourth field LEN is expected to be
304 the number of fields - 1, not the total number of fields. */
305 finish_builtin_type (t, "cp_eh_info", fields, 6, ptr_type_node);
306 t = build_pointer_type (t);
307
308 /* And now the function. */
309 fn = build_lang_decl (FUNCTION_DECL, fn,
310 build_function_type (t, void_list_node));
311 DECL_EXTERNAL (fn) = 1;
312 TREE_PUBLIC (fn) = 1;
313 DECL_ARTIFICIAL (fn) = 1;
314 pushdecl_top_level (fn);
315 make_function_rtl (fn);
316 assemble_external (fn);
317 pop_obstacks ();
318 }
319 return build_function_call (fn, NULL_TREE);
320 }
321
322 /* Retrieve a pointer to the cp_eh_info node for the current exception
323 and save it in the current binding level. */
324
325 static void
326 push_eh_info ()
327 {
328 tree decl, fn = call_eh_info ();
329
330 /* Remember the pointer to the current exception info; it won't change
331 during this catch block. */
332 decl = build_decl (VAR_DECL, get_identifier ("__exception_info"),
333 TREE_TYPE (fn));
334 DECL_ARTIFICIAL (decl) = 1;
335 DECL_INITIAL (decl) = fn;
336 decl = pushdecl (decl);
337 cp_finish_decl (decl, fn, NULL_TREE, 0, 0);
338 }
339
340 /* Returns a reference to the cp_eh_info node for the current exception. */
341
342 static tree
343 get_eh_info ()
344 {
345 /* Look for the pointer pushed in push_eh_info. */
346 tree t = lookup_name (get_identifier ("__exception_info"), 0);
347 return build_indirect_ref (t, NULL_PTR);
348 }
349
350 /* Returns a reference to the current exception object. */
351
352 static tree
353 get_eh_value ()
354 {
355 return build_component_ref (get_eh_info (), get_identifier ("value"),
356 NULL_TREE, 0);
357 }
358
359 /* Returns a reference to the current exception type. */
360
361 static tree
362 get_eh_type ()
363 {
364 return build_component_ref (get_eh_info (), get_identifier ("type"),
365 NULL_TREE, 0);
366 }
367
368 /* Returns a reference to whether or not the current exception
369 has been caught. */
370
371 static tree
372 get_eh_caught ()
373 {
374 return build_component_ref (get_eh_info (), get_identifier ("caught"),
375 NULL_TREE, 0);
376 }
377
378 /* Returns a reference to whether or not the current exception
379 has been caught. */
380
381 static tree
382 get_eh_handlers ()
383 {
384 return build_component_ref (get_eh_info (), get_identifier ("handlers"),
385 NULL_TREE, 0);
386 }
387
388 /* Build a type value for use at runtime for a type that is matched
389 against by the exception handling system. */
390
391 static tree
392 build_eh_type_type (type)
393 tree type;
394 {
395 char *typestring;
396 tree exp;
397
398 if (type == error_mark_node)
399 return error_mark_node;
400
401 /* peel back references, so they match. */
402 if (TREE_CODE (type) == REFERENCE_TYPE)
403 type = TREE_TYPE (type);
404
405 /* Peel off cv qualifiers. */
406 type = TYPE_MAIN_VARIANT (type);
407
408 if (flag_rtti)
409 {
410 return build1 (ADDR_EXPR, ptr_type_node, get_typeid (type));
411 }
412
413 typestring = build_overload_name (type, 1, 1);
414 exp = combine_strings (build_string (strlen (typestring)+1, typestring));
415 return build1 (ADDR_EXPR, ptr_type_node, exp);
416 }
417
418 /* Build the address of a runtime type for use in the runtime matching
419 field of the new exception model */
420
421 static tree
422 build_eh_type_type_ref (type)
423 tree type;
424 {
425 char *typestring;
426 tree exp;
427 int susp;
428
429 if (type == error_mark_node)
430 return error_mark_node;
431
432 /* peel back references, so they match. */
433 if (TREE_CODE (type) == REFERENCE_TYPE)
434 type = TREE_TYPE (type);
435
436 /* Peel off cv qualifiers. */
437 type = TYPE_MAIN_VARIANT (type);
438
439 push_obstacks_nochange ();
440 end_temporary_allocation ();
441
442 if (flag_rtti)
443 {
444 exp = get_tinfo_fn (type);
445 TREE_USED (exp) = 1;
446 mark_inline_for_output (exp);
447 exp = build1 (ADDR_EXPR, ptr_type_node, exp);
448 }
449 else
450 {
451 typestring = build_overload_name (type, 1, 1);
452 exp = combine_strings (build_string (strlen (typestring)+1, typestring));
453 exp = build1 (ADDR_EXPR, ptr_type_node, exp);
454 }
455 pop_obstacks ();
456 return (exp);
457 }
458
459
460 /* Build a type value for use at runtime for a exp that is thrown or
461 matched against by the exception handling system. */
462
463 static tree
464 build_eh_type (exp)
465 tree exp;
466 {
467 if (flag_rtti)
468 {
469 exp = build_typeid (exp);
470 return build1 (ADDR_EXPR, ptr_type_node, exp);
471 }
472 return build_eh_type_type (TREE_TYPE (exp));
473 }
474
475 /* This routine is called to mark all the symbols representing runtime
476 type functions in the exception table as haveing been referenced.
477 This will make sure code is emitted for them. Called from finish_file. */
478 void
479 mark_all_runtime_matches ()
480 {
481 int x,num;
482 void **ptr;
483 tree exp;
484
485 num = find_all_handler_type_matches (&ptr);
486 if (num == 0 || ptr == NULL)
487 return;
488
489 for (x=0; x <num; x++)
490 {
491 exp = (tree) ptr[x];
492 if (TREE_CODE (exp) == ADDR_EXPR)
493 {
494 exp = TREE_OPERAND (exp, 0);
495 if (TREE_CODE (exp) == FUNCTION_DECL)
496 TREE_SYMBOL_REFERENCED (DECL_ASSEMBLER_NAME (exp)) = 1;
497 }
498 }
499
500 free (ptr);
501 }
502
503 /* Build up a call to __cp_pop_exception, to destroy the exception object
504 for the current catch block. HANDLER is either true or false, telling
505 the library whether or not it is being called from an exception handler;
506 if it is, it avoids destroying the object on rethrow. */
507
508 static tree
509 do_pop_exception ()
510 {
511 tree fn, cleanup;
512 fn = get_identifier ("__cp_pop_exception");
513 if (IDENTIFIER_GLOBAL_VALUE (fn))
514 fn = IDENTIFIER_GLOBAL_VALUE (fn);
515 else
516 {
517 /* Declare void __cp_pop_exception (void *),
518 as defined in exception.cc. */
519 push_obstacks_nochange ();
520 end_temporary_allocation ();
521 fn = build_lang_decl
522 (FUNCTION_DECL, fn,
523 build_function_type (void_type_node, tree_cons
524 (NULL_TREE, ptr_type_node, void_list_node)));
525 DECL_EXTERNAL (fn) = 1;
526 TREE_PUBLIC (fn) = 1;
527 DECL_ARTIFICIAL (fn) = 1;
528 pushdecl_top_level (fn);
529 make_function_rtl (fn);
530 assemble_external (fn);
531 pop_obstacks ();
532 }
533
534 /* Arrange to do a dynamically scoped cleanup upon exit from this region. */
535 cleanup = lookup_name (get_identifier ("__exception_info"), 0);
536 cleanup = build_function_call (fn, expr_tree_cons
537 (NULL_TREE, cleanup, NULL_TREE));
538 return cleanup;
539 }
540
541 /* This routine creates the cleanup for the current exception. */
542
543 static void
544 push_eh_cleanup ()
545 {
546 int yes;
547
548 expand_expr (build_unary_op (PREINCREMENT_EXPR, get_eh_handlers (), 1),
549 const0_rtx, VOIDmode, EXPAND_NORMAL);
550
551 yes = suspend_momentary ();
552 /* All cleanups must last longer than normal. */
553 expand_decl_cleanup (NULL_TREE, do_pop_exception ());
554 resume_momentary (yes);
555 }
556
557 /* Build up a call to terminate on the function obstack, for use as an
558 exception handler. */
559
560 tree
561 build_terminate_handler ()
562 {
563 int yes = suspend_momentary ();
564 tree term = build_function_call (Terminate, NULL_TREE);
565 resume_momentary (yes);
566 return term;
567 }
568
569 /* Call this to start a catch block. Typename is the typename, and identifier
570 is the variable to place the object in or NULL if the variable doesn't
571 matter. If typename is NULL, that means its a "catch (...)" or catch
572 everything. In that case we don't need to do any type checking.
573 (ie: it ends up as the "else" clause rather than an "else if" clause) */
574
575 void
576 expand_start_catch_block (declspecs, declarator)
577 tree declspecs, declarator;
578 {
579 tree decl;
580
581 if (processing_template_decl)
582 {
583 if (declspecs)
584 {
585 decl = grokdeclarator (declarator, declspecs, CATCHPARM,
586 1, NULL_TREE);
587 pushdecl (decl);
588 decl = build_min_nt (DECL_STMT, copy_to_permanent (declarator),
589 copy_to_permanent (declspecs),
590 NULL_TREE);
591 add_tree (decl);
592 }
593 return;
594 }
595
596 if (! doing_eh (1))
597 return;
598
599 if (flag_new_exceptions)
600 process_start_catch_block (declspecs, declarator);
601 else
602 process_start_catch_block_old (declspecs, declarator);
603 }
604
605
606 /* This function performs the expand_start_catch_block functionality for
607 exceptions implemented in the old style, where catch blocks were all
608 called, and had to check the runtime information themselves. */
609
610 static void
611 process_start_catch_block_old (declspecs, declarator)
612 tree declspecs, declarator;
613 {
614 rtx false_label_rtx;
615 tree decl = NULL_TREE;
616 tree init;
617
618 /* Create a binding level for the eh_info and the exception object
619 cleanup. */
620 pushlevel (0);
621 expand_start_bindings (0);
622
623 false_label_rtx = gen_label_rtx ();
624 push_label_entry (&false_label_stack, false_label_rtx, NULL_TREE);
625
626 emit_line_note (input_filename, lineno);
627
628 push_eh_info ();
629
630 if (declspecs)
631 {
632 decl = grokdeclarator (declarator, declspecs, CATCHPARM, 1, NULL_TREE);
633
634 if (decl == NULL_TREE)
635 error ("invalid catch parameter");
636 }
637
638 if (decl)
639 {
640 tree exp;
641 rtx call_rtx, return_value_rtx;
642 tree init_type;
643
644 /* Make sure we mark the catch param as used, otherwise we'll get
645 a warning about an unused ((anonymous)). */
646 TREE_USED (decl) = 1;
647
648 /* Figure out the type that the initializer is. */
649 init_type = TREE_TYPE (decl);
650 if (TREE_CODE (init_type) != REFERENCE_TYPE
651 && TREE_CODE (init_type) != POINTER_TYPE)
652 init_type = build_reference_type (init_type);
653
654 exp = get_eh_value ();
655
656 /* Since pointers are passed by value, initialize a reference to
657 pointer catch parm with the address of the value slot. */
658 if (TREE_CODE (init_type) == REFERENCE_TYPE
659 && TREE_CODE (TREE_TYPE (init_type)) == POINTER_TYPE)
660 exp = build_unary_op (ADDR_EXPR, exp, 1);
661
662 exp = expr_tree_cons (NULL_TREE,
663 build_eh_type_type (TREE_TYPE (decl)),
664 expr_tree_cons (NULL_TREE,
665 get_eh_type (),
666 expr_tree_cons (NULL_TREE, exp, NULL_TREE)));
667 exp = build_function_call (CatchMatch, exp);
668 call_rtx = expand_call (exp, NULL_RTX, 0);
669
670 return_value_rtx = hard_function_value (ptr_type_node, exp);
671
672 /* did the throw type match function return TRUE? */
673 emit_cmp_insn (return_value_rtx, const0_rtx, EQ, NULL_RTX,
674 GET_MODE (return_value_rtx), 0, 0);
675
676 /* if it returned FALSE, jump over the catch block, else fall into it */
677 emit_jump_insn (gen_beq (false_label_rtx));
678
679 push_eh_cleanup ();
680
681 /* Create a binding level for the parm. */
682 pushlevel (0);
683 expand_start_bindings (0);
684
685 init = convert_from_reference (make_tree (init_type, call_rtx));
686
687 /* If the constructor for the catch parm exits via an exception, we
688 must call terminate. See eh23.C. */
689 if (TYPE_NEEDS_CONSTRUCTING (TREE_TYPE (decl)))
690 {
691 /* Generate the copy constructor call directly so we can wrap it.
692 See also expand_default_init. */
693 init = ocp_convert (TREE_TYPE (decl), init,
694 CONV_IMPLICIT|CONV_FORCE_TEMP, 0);
695 init = build (TRY_CATCH_EXPR, TREE_TYPE (init), init,
696 build_terminate_handler ());
697 }
698
699 /* Let `cp_finish_decl' know that this initializer is ok. */
700 DECL_INITIAL (decl) = init;
701 decl = pushdecl (decl);
702
703 cp_finish_decl (decl, init, NULL_TREE, 0, LOOKUP_ONLYCONVERTING);
704 }
705 else
706 {
707 push_eh_cleanup ();
708
709 /* Create a binding level for the parm. */
710 pushlevel (0);
711 expand_start_bindings (0);
712
713 /* Fall into the catch all section. */
714 }
715
716 init = build_modify_expr (get_eh_caught (), NOP_EXPR, integer_one_node);
717 expand_expr (init, const0_rtx, VOIDmode, EXPAND_NORMAL);
718
719 emit_line_note (input_filename, lineno);
720 }
721
722 /* This function performs the expand_start_catch_block functionality for
723 exceptions implemented in the new style. __throw determines whether
724 a handler needs to be called or not, so the handler itself has to do
725 nothing additionaal. */
726
727 static void
728 process_start_catch_block (declspecs, declarator)
729 tree declspecs, declarator;
730 {
731 rtx false_label_rtx;
732 tree decl = NULL_TREE;
733 tree init;
734
735 /* Create a binding level for the eh_info and the exception object
736 cleanup. */
737 pushlevel (0);
738 expand_start_bindings (0);
739
740
741 if (declspecs)
742 {
743 decl = grokdeclarator (declarator, declspecs, CATCHPARM, 1, NULL_TREE);
744
745 if (decl == NULL_TREE)
746 error ("invalid catch parameter");
747 }
748
749 if (decl)
750 start_catch_handler (build_eh_type_type_ref (TREE_TYPE (decl)));
751 else
752 start_catch_handler (CATCH_ALL_TYPE);
753
754 emit_line_note (input_filename, lineno);
755
756 push_eh_info ();
757
758 if (decl)
759 {
760 tree exp;
761 rtx call_rtx, return_value_rtx;
762 tree init_type;
763
764 /* Make sure we mark the catch param as used, otherwise we'll get
765 a warning about an unused ((anonymous)). */
766 TREE_USED (decl) = 1;
767
768 /* Figure out the type that the initializer is. */
769 init_type = TREE_TYPE (decl);
770 if (TREE_CODE (init_type) != REFERENCE_TYPE
771 && TREE_CODE (init_type) != POINTER_TYPE)
772 init_type = build_reference_type (init_type);
773
774 exp = get_eh_value ();
775
776 /* Since pointers are passed by value, initialize a reference to
777 pointer catch parm with the address of the value slot. */
778 if (TREE_CODE (init_type) == REFERENCE_TYPE
779 && TREE_CODE (TREE_TYPE (init_type)) == POINTER_TYPE)
780 exp = build_unary_op (ADDR_EXPR, exp, 1);
781
782 exp = ocp_convert (init_type , exp, CONV_IMPLICIT|CONV_FORCE_TEMP, 0);
783
784 push_eh_cleanup ();
785
786 /* Create a binding level for the parm. */
787 pushlevel (0);
788 expand_start_bindings (0);
789
790 init = convert_from_reference (exp);
791
792 /* If the constructor for the catch parm exits via an exception, we
793 must call terminate. See eh23.C. */
794 if (TYPE_NEEDS_CONSTRUCTING (TREE_TYPE (decl)))
795 {
796 /* Generate the copy constructor call directly so we can wrap it.
797 See also expand_default_init. */
798 init = ocp_convert (TREE_TYPE (decl), init,
799 CONV_IMPLICIT|CONV_FORCE_TEMP, 0);
800 init = build (TRY_CATCH_EXPR, TREE_TYPE (init), init,
801 build_terminate_handler ());
802 }
803
804 /* Let `cp_finish_decl' know that this initializer is ok. */
805 DECL_INITIAL (decl) = init;
806 decl = pushdecl (decl);
807
808 cp_finish_decl (decl, init, NULL_TREE, 0, LOOKUP_ONLYCONVERTING);
809 }
810 else
811 {
812 push_eh_cleanup ();
813
814 /* Create a binding level for the parm. */
815 pushlevel (0);
816 expand_start_bindings (0);
817
818 /* Fall into the catch all section. */
819 }
820
821 init = build_modify_expr (get_eh_caught (), NOP_EXPR, integer_one_node);
822 expand_expr (init, const0_rtx, VOIDmode, EXPAND_NORMAL);
823
824 emit_line_note (input_filename, lineno);
825 }
826
827
828
829 /* Call this to end a catch block. Its responsible for emitting the
830 code to handle jumping back to the correct place, and for emitting
831 the label to jump to if this catch block didn't match. */
832
833 void
834 expand_end_catch_block ()
835 {
836 if (! doing_eh (1))
837 return;
838
839 /* Cleanup the EH parameter. */
840 expand_end_bindings (getdecls (), kept_level_p (), 0);
841 poplevel (kept_level_p (), 1, 0);
842
843 /* Cleanup the EH object. */
844 expand_end_bindings (getdecls (), kept_level_p (), 0);
845 poplevel (kept_level_p (), 1, 0);
846
847 /* Fall to outside the try statement when done executing handler and
848 we fall off end of handler. This is jump Lresume in the
849 documentation. */
850 expand_goto (top_label_entry (&caught_return_label_stack));
851
852 /* label we emit to jump to if this catch block didn't match. */
853 /* This the closing } in the `if (eq) {' of the documentation. */
854 if (! flag_new_exceptions)
855 emit_label (pop_label_entry (&false_label_stack));
856 }
857
858 /* An exception spec is implemented more or less like:
859
860 try {
861 function body;
862 } catch (...) {
863 void *p[] = { typeid(raises) };
864 __check_eh_spec (p, count);
865 }
866
867 __check_eh_spec in exception.cc handles all the details. */
868
869 void
870 expand_start_eh_spec ()
871 {
872 expand_start_try_stmts ();
873 }
874
875 static void
876 expand_end_eh_spec (raises)
877 tree raises;
878 {
879 tree tmp, fn, decl, types = NULL_TREE;
880 int count = 0;
881
882 expand_start_all_catch ();
883 expand_start_catch_block (NULL_TREE, NULL_TREE);
884
885 /* Build up an array of type_infos. */
886 for (; raises && TREE_VALUE (raises); raises = TREE_CHAIN (raises))
887 {
888 types = expr_tree_cons
889 (NULL_TREE, build_eh_type_type (TREE_VALUE (raises)), types);
890 ++count;
891 }
892
893 types = build_nt (CONSTRUCTOR, NULL_TREE, types);
894 TREE_HAS_CONSTRUCTOR (types) = 1;
895
896 /* We can't pass the CONSTRUCTOR directly, so stick it in a variable. */
897 tmp = build_array_type (const_ptr_type_node, NULL_TREE);
898 decl = build_decl (VAR_DECL, NULL_TREE, tmp);
899 DECL_ARTIFICIAL (decl) = 1;
900 DECL_INITIAL (decl) = types;
901 cp_finish_decl (decl, types, NULL_TREE, 0, 0);
902
903 decl = decay_conversion (decl);
904
905 fn = get_identifier ("__check_eh_spec");
906 if (IDENTIFIER_GLOBAL_VALUE (fn))
907 fn = IDENTIFIER_GLOBAL_VALUE (fn);
908 else
909 {
910 push_obstacks_nochange ();
911 end_temporary_allocation ();
912
913 tmp = tree_cons
914 (NULL_TREE, integer_type_node, tree_cons
915 (NULL_TREE, TREE_TYPE (decl), void_list_node));
916 tmp = build_function_type (void_type_node, tmp);
917
918 fn = build_lang_decl (FUNCTION_DECL, fn, tmp);
919 DECL_EXTERNAL (fn) = 1;
920 TREE_PUBLIC (fn) = 1;
921 DECL_ARTIFICIAL (fn) = 1;
922 TREE_THIS_VOLATILE (fn) = 1;
923 pushdecl_top_level (fn);
924 make_function_rtl (fn);
925 assemble_external (fn);
926 pop_obstacks ();
927 }
928
929 tmp = expr_tree_cons (NULL_TREE, build_int_2 (count, 0), expr_tree_cons
930 (NULL_TREE, decl, NULL_TREE));
931 tmp = build_call (fn, TREE_TYPE (TREE_TYPE (fn)), tmp);
932 expand_expr (tmp, const0_rtx, VOIDmode, EXPAND_NORMAL);
933
934 expand_end_catch_block ();
935 expand_end_all_catch ();
936 }
937
938 /* This is called to expand all the toplevel exception handling
939 finalization for a function. It should only be called once per
940 function. */
941
942 void
943 expand_exception_blocks ()
944 {
945 do_pending_stack_adjust ();
946 push_to_sequence (catch_clauses);
947 expand_leftover_cleanups ();
948 do_pending_stack_adjust ();
949 catch_clauses = get_insns ();
950 end_sequence ();
951
952 /* Do this after we expand leftover cleanups, so that the
953 expand_eh_region_end that expand_end_eh_spec does will match the
954 right expand_eh_region_start, and make sure it comes out before
955 the terminate protected region. */
956 if (TYPE_RAISES_EXCEPTIONS (TREE_TYPE (current_function_decl)))
957 {
958 expand_end_eh_spec (TYPE_RAISES_EXCEPTIONS (TREE_TYPE (current_function_decl)));
959 do_pending_stack_adjust ();
960 push_to_sequence (catch_clauses);
961 expand_leftover_cleanups ();
962 do_pending_stack_adjust ();
963 catch_clauses = get_insns ();
964 end_sequence ();
965 }
966
967 if (catch_clauses)
968 {
969 rtx funcend = gen_label_rtx ();
970 emit_jump (funcend);
971
972 /* We cannot protect n regions this way if we must flow into the
973 EH region through the top of the region, as we have to with
974 the setjmp/longjmp approach. */
975 if (exceptions_via_longjmp == 0)
976 expand_eh_region_start ();
977
978 emit_insns (catch_clauses);
979 catch_clauses = NULL_RTX;
980
981 if (exceptions_via_longjmp == 0)
982 expand_eh_region_end (build_terminate_handler ());
983
984 expand_leftover_cleanups ();
985
986 emit_label (funcend);
987 }
988 }
989
990 tree
991 start_anon_func ()
992 {
993 static int counter = 0;
994 int old_interface_unknown = interface_unknown;
995 char name[32];
996 tree params;
997 tree t;
998
999 push_cp_function_context (NULL_TREE);
1000 push_to_top_level ();
1001
1002 /* No need to mangle this. */
1003 push_lang_context (lang_name_c);
1004
1005 interface_unknown = 1;
1006
1007 params = void_list_node;
1008 /* tcf stands for throw clean function. */
1009 sprintf (name, "__tcf_%d", counter++);
1010 t = make_call_declarator (get_identifier (name), params, NULL_TREE,
1011 NULL_TREE);
1012 start_function (decl_tree_cons (NULL_TREE, get_identifier ("static"),
1013 void_list_node),
1014 t, NULL_TREE, 0);
1015 store_parm_decls ();
1016 pushlevel (0);
1017 clear_last_expr ();
1018 push_momentary ();
1019 expand_start_bindings (0);
1020 emit_line_note (input_filename, lineno);
1021
1022 interface_unknown = old_interface_unknown;
1023
1024 pop_lang_context ();
1025
1026 return current_function_decl;
1027 }
1028
1029 void
1030 end_anon_func ()
1031 {
1032 expand_end_bindings (getdecls (), 1, 0);
1033 poplevel (1, 0, 0);
1034 pop_momentary ();
1035
1036 finish_function (lineno, 0, 0);
1037
1038 pop_from_top_level ();
1039 pop_cp_function_context (NULL_TREE);
1040 }
1041
1042 /* Return a pointer to a buffer for an exception object of type TYPE. */
1043
1044 tree
1045 alloc_eh_object (type)
1046 tree type;
1047 {
1048 tree fn, exp;
1049
1050 fn = get_identifier ("__eh_alloc");
1051 if (IDENTIFIER_GLOBAL_VALUE (fn))
1052 fn = IDENTIFIER_GLOBAL_VALUE (fn);
1053 else
1054 {
1055 /* Declare __eh_alloc (size_t), as defined in exception.cc. */
1056 tree tmp;
1057 push_obstacks_nochange ();
1058 end_temporary_allocation ();
1059 tmp = tree_cons (NULL_TREE, sizetype, void_list_node);
1060 fn = build_lang_decl (FUNCTION_DECL, fn,
1061 build_function_type (ptr_type_node, tmp));
1062 DECL_EXTERNAL (fn) = 1;
1063 TREE_PUBLIC (fn) = 1;
1064 DECL_ARTIFICIAL (fn) = 1;
1065 pushdecl_top_level (fn);
1066 make_function_rtl (fn);
1067 assemble_external (fn);
1068 pop_obstacks ();
1069 }
1070
1071 exp = build_function_call (fn, expr_tree_cons
1072 (NULL_TREE, size_in_bytes (type), NULL_TREE));
1073 exp = build1 (NOP_EXPR, build_pointer_type (type), exp);
1074 return exp;
1075 }
1076
1077 /* Expand a throw statement. This follows the following
1078 algorithm:
1079
1080 1. Allocate space to save the current PC onto the stack.
1081 2. Generate and emit a label and save its address into the
1082 newly allocated stack space since we can't save the pc directly.
1083 3. If this is the first call to throw in this function:
1084 generate a label for the throw block
1085 4. jump to the throw block label. */
1086
1087 void
1088 expand_throw (exp)
1089 tree exp;
1090 {
1091 tree fn;
1092 static tree cleanup_type;
1093
1094 if (! doing_eh (1))
1095 return;
1096
1097 if (exp)
1098 {
1099 tree throw_type;
1100 tree cleanup = NULL_TREE, e;
1101
1102 /* throw expression */
1103 /* First, decay it. */
1104 exp = decay_conversion (exp);
1105
1106 /* cleanup_type is void (*)(void *, int),
1107 the internal type of a destructor. */
1108 if (cleanup_type == NULL_TREE)
1109 {
1110 push_obstacks_nochange ();
1111 end_temporary_allocation ();
1112 cleanup_type = build_pointer_type
1113 (build_function_type
1114 (void_type_node, tree_cons
1115 (NULL_TREE, ptr_type_node, tree_cons
1116 (NULL_TREE, integer_type_node, void_list_node))));
1117 pop_obstacks ();
1118 }
1119
1120 if (TREE_CODE (TREE_TYPE (exp)) == POINTER_TYPE)
1121 {
1122 throw_type = build_eh_type (exp);
1123 exp = build_reinterpret_cast (ptr_type_node, exp);
1124 }
1125 else
1126 {
1127 tree object, ptr;
1128
1129 /* OK, this is kind of wacky. The WP says that we call
1130 terminate
1131
1132 when the exception handling mechanism, after completing
1133 evaluation of the expression to be thrown but before the
1134 exception is caught (_except.throw_), calls a user function
1135 that exits via an uncaught exception.
1136
1137 So we have to protect the actual initialization of the
1138 exception object with terminate(), but evaluate the expression
1139 first. We also expand the call to __eh_alloc
1140 first. Since there could be temps in the expression, we need
1141 to handle that, too. */
1142
1143 expand_start_target_temps ();
1144
1145 #if 0
1146 /* Unfortunately, this doesn't work. */
1147 preexpand_calls (exp);
1148 #else
1149 /* Store the throw expression into a temp. This can be less
1150 efficient than storing it into the allocated space directly, but
1151 oh well. To do this efficiently we would need to insinuate
1152 ourselves into expand_call. */
1153 if (TREE_SIDE_EFFECTS (exp))
1154 {
1155 tree temp = build (VAR_DECL, TREE_TYPE (exp));
1156 DECL_ARTIFICIAL (temp) = 1;
1157 layout_decl (temp, 0);
1158 DECL_RTL (temp) = assign_temp (TREE_TYPE (exp), 2, 0, 1);
1159 expand_expr (build (INIT_EXPR, TREE_TYPE (exp), temp, exp),
1160 NULL_RTX, VOIDmode, 0);
1161 expand_decl_cleanup (NULL_TREE, maybe_build_cleanup (temp));
1162 exp = temp;
1163 }
1164 #endif
1165
1166 /* Allocate the space for the exception. */
1167 ptr = save_expr (alloc_eh_object (TREE_TYPE (exp)));
1168 expand_expr (ptr, const0_rtx, VOIDmode, 0);
1169
1170 expand_eh_region_start ();
1171
1172 object = build_indirect_ref (ptr, NULL_PTR);
1173 exp = build_modify_expr (object, INIT_EXPR, exp);
1174
1175 if (exp == error_mark_node)
1176 error (" in thrown expression");
1177
1178 expand_expr (exp, const0_rtx, VOIDmode, 0);
1179 expand_eh_region_end (build_terminate_handler ());
1180 expand_end_target_temps ();
1181
1182 throw_type = build_eh_type (object);
1183
1184 if (TYPE_HAS_DESTRUCTOR (TREE_TYPE (object)))
1185 {
1186 cleanup = lookup_fnfields (TYPE_BINFO (TREE_TYPE (object)),
1187 dtor_identifier, 0);
1188 cleanup = TREE_VALUE (cleanup);
1189 mark_used (cleanup);
1190 mark_addressable (cleanup);
1191 /* Pretend it's a normal function. */
1192 cleanup = build1 (ADDR_EXPR, cleanup_type, cleanup);
1193 }
1194
1195 exp = ptr;
1196 }
1197
1198 if (cleanup == NULL_TREE)
1199 {
1200 cleanup = build_int_2 (0, 0);
1201 TREE_TYPE (cleanup) = cleanup_type;
1202 }
1203
1204 fn = get_identifier ("__cp_push_exception");
1205 if (IDENTIFIER_GLOBAL_VALUE (fn))
1206 fn = IDENTIFIER_GLOBAL_VALUE (fn);
1207 else
1208 {
1209 /* Declare __cp_push_exception (void*, void*, void (*)(void*, int)),
1210 as defined in exception.cc. */
1211 tree tmp;
1212 push_obstacks_nochange ();
1213 end_temporary_allocation ();
1214 tmp = tree_cons
1215 (NULL_TREE, ptr_type_node, tree_cons
1216 (NULL_TREE, ptr_type_node, tree_cons
1217 (NULL_TREE, cleanup_type, void_list_node)));
1218 fn = build_lang_decl (FUNCTION_DECL, fn,
1219 build_function_type (void_type_node, tmp));
1220 DECL_EXTERNAL (fn) = 1;
1221 TREE_PUBLIC (fn) = 1;
1222 DECL_ARTIFICIAL (fn) = 1;
1223 pushdecl_top_level (fn);
1224 make_function_rtl (fn);
1225 assemble_external (fn);
1226 pop_obstacks ();
1227 }
1228
1229 e = expr_tree_cons (NULL_TREE, exp, expr_tree_cons
1230 (NULL_TREE, throw_type, expr_tree_cons
1231 (NULL_TREE, cleanup, NULL_TREE)));
1232 e = build_function_call (fn, e);
1233 expand_expr (e, const0_rtx, VOIDmode, 0);
1234 }
1235 else
1236 {
1237 /* rethrow current exception; note that it's no longer caught. */
1238
1239 tree fn = get_identifier ("__uncatch_exception");
1240 if (IDENTIFIER_GLOBAL_VALUE (fn))
1241 fn = IDENTIFIER_GLOBAL_VALUE (fn);
1242 else
1243 {
1244 /* Declare void __uncatch_exception (void)
1245 as defined in exception.cc. */
1246 push_obstacks_nochange ();
1247 end_temporary_allocation ();
1248 fn = build_lang_decl (FUNCTION_DECL, fn,
1249 build_function_type (void_type_node,
1250 void_list_node));
1251 DECL_EXTERNAL (fn) = 1;
1252 TREE_PUBLIC (fn) = 1;
1253 DECL_ARTIFICIAL (fn) = 1;
1254 pushdecl_top_level (fn);
1255 make_function_rtl (fn);
1256 assemble_external (fn);
1257 pop_obstacks ();
1258 }
1259
1260 exp = build_function_call (fn, NULL_TREE);
1261 expand_expr (exp, const0_rtx, VOIDmode, EXPAND_NORMAL);
1262 }
1263
1264 expand_internal_throw ();
1265 }
1266
1267 /* Build a throw expression. */
1268
1269 tree
1270 build_throw (e)
1271 tree e;
1272 {
1273 if (e == error_mark_node)
1274 return e;
1275
1276 if (processing_template_decl)
1277 return build_min (THROW_EXPR, void_type_node, e);
1278
1279 if (! flag_ansi && e == null_node)
1280 {
1281 cp_warning ("throwing NULL");
1282 e = integer_zero_node;
1283 }
1284
1285 e = build1 (THROW_EXPR, void_type_node, e);
1286 TREE_SIDE_EFFECTS (e) = 1;
1287 TREE_USED (e) = 1;
1288
1289 return e;
1290 }
This page took 0.097735 seconds and 5 git commands to generate.