]> gcc.gnu.org Git - gcc.git/blame - gcc/cp/except.c
[multiple changes]
[gcc.git] / gcc / cp / except.c
CommitLineData
8d08fdba 1/* Handle exceptional things in C++.
e5e809f4 2 Copyright (C) 1989, 92-97, 1998 Free Software Foundation, Inc.
8d2733ca
MS
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.
8d08fdba
MS
6
7This file is part of GNU CC.
8
9GNU CC is free software; you can redistribute it and/or modify
10it under the terms of the GNU General Public License as published by
11the Free Software Foundation; either version 2, or (at your option)
12any later version.
13
14GNU CC is distributed in the hope that it will be useful,
15but WITHOUT ANY WARRANTY; without even the implied warranty of
16MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
17GNU General Public License for more details.
18
19You should have received a copy of the GNU General Public License
20along with GNU CC; see the file COPYING. If not, write to
e9fa0c7c
RK
21the Free Software Foundation, 59 Temple Place - Suite 330,
22Boston, MA 02111-1307, USA. */
8d08fdba
MS
23
24
8d08fdba 25#include "config.h"
8d052bc7 26#include "system.h"
8d08fdba
MS
27#include "tree.h"
28#include "rtl.h"
29#include "cp-tree.h"
30#include "flags.h"
8d2733ca
MS
31#include "obstack.h"
32#include "expr.h"
21451173 33#include "output.h"
6467930b
MS
34#include "except.h"
35#include "function.h"
0021b564 36#include "defaults.h"
54f92bfb 37#include "toplev.h"
9a0d1e1b 38#include "eh-common.h"
8d08fdba 39
f30432d7 40rtx expand_builtin_return_addr PROTO((enum built_in_function, int, rtx));
a3b49ccd 41
eb66be0e 42/* Holds the fndecl for __builtin_return_address. */
8d2733ca 43tree builtin_return_address_fndecl;
8d2733ca
MS
44
45/* A couple of backend routines from m88k.c */
46
49c249e1 47static void push_eh_cleanup PROTO((void));
49c249e1
JM
48static tree build_eh_type_type PROTO((tree));
49static tree build_eh_type PROTO((tree));
50static void expand_end_eh_spec PROTO((tree));
69ac77ce
JL
51static tree call_eh_info PROTO((void));
52static void push_eh_info PROTO((void));
53static tree get_eh_info PROTO((void));
54static tree get_eh_value PROTO((void));
55static tree get_eh_type PROTO((void));
56static tree get_eh_caught PROTO((void));
57static tree get_eh_handlers PROTO((void));
58static tree do_pop_exception PROTO((void));
5816cb14
AM
59static void process_start_catch_block PROTO((tree, tree));
60static void process_start_catch_block_old PROTO((tree, tree));
61static tree build_eh_type_type_ref PROTO((tree));
8d2733ca 62
8d2733ca 63#if 0
e92cc029 64/* This is the startup, and finish stuff per exception table. */
8d08fdba 65
8d2733ca
MS
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
8d08fdba 70
8d2733ca 71#ifdef EXCEPT_SECTION_ASM_OP
8d08fdba 72
8d2733ca
MS
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! */
76asm (EXCEPT_SECTION_ASM_OP);
77exception_table __EXCEPTION_TABLE__[1] = { (void*)0, (void*)0, (void*)0 };
78asm (TEXT_SECTION_ASM_OP);
8d08fdba 79
8d2733ca 80#endif /* EXCEPT_SECTION_ASM_OP */
8d08fdba 81
8d2733ca
MS
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
87asm (EXCEPT_SECTION_ASM_OP);
88exception_table __EXCEPTION_END__[1] = { (void*)-1, (void*)-1, (void*)-1 };
89asm (TEXT_SECTION_ASM_OP);
90
91#endif /* EXCEPT_SECTION_ASM_OP */
92
93#endif
8d08fdba 94
8d2733ca 95#include "decl.h"
8d2733ca
MS
96#include "insn-flags.h"
97#include "obstack.h"
8d2733ca
MS
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
6467930b
MS
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
8d2733ca
MS
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
6467930b 114 in the finalization variable in the eh_entry structure. Otherwise
8d2733ca 115 the finalization variable is set to NULL to reflect the fact that
38e01259 116 it is the end of a try block. Next, this modified eh_entry node
8d2733ca
MS
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
f30432d7
MS
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
8d2733ca
MS
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
6467930b 145 entry in the eh_queue until we get to an entry with a NULL
8d2733ca
MS
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
6467930b 150 enqueue'ing this new eh_entry to be output at an outer level.
8d2733ca
MS
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
8d2733ca
MS
158/* local globals for function calls
159 ====================================================================== */
160
fb98cff6
JM
161/* Used to cache "terminate" and "__throw_type_match*". */
162static tree Terminate, CatchMatch;
8d2733ca 163
eb66be0e 164/* Used to cache __find_first_exception_table_match for throw. */
8d2733ca
MS
165static tree FirstExceptionMatch;
166
eb66be0e 167/* Used to cache a call to __unwind_function. */
8d2733ca
MS
168static tree Unwind;
169
8d2733ca
MS
170/* ====================================================================== */
171
172
8d2733ca
MS
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
6467930b 182extern rtx catch_clauses;
6c20b7e9 183extern tree const_ptr_type_node;
8d2733ca 184
8d2733ca
MS
185/* ========================================================================= */
186
8d2733ca
MS
187/* sets up all the global eh stuff that needs to be initialized at the
188 start of compilation.
189
190 This includes:
6467930b 191 - Setting up all the function call trees. */
8d08fdba 192
8d08fdba 193void
8d2733ca 194init_exception_processing ()
8d08fdba 195{
6633d636
MS
196 /* void vtype () */
197 tree vtype = build_function_type (void_type_node, void_list_node);
198
2c73f9f5
ML
199 if (flag_honor_std)
200 push_namespace (get_identifier ("std"));
fb98cff6
JM
201 Terminate = auto_function (get_identifier ("terminate"),
202 vtype, NOT_BUILT_IN);
203 TREE_THIS_VOLATILE (Terminate) = 1;
2c73f9f5
ML
204 if (flag_honor_std)
205 pop_namespace ();
8ccc31eb 206
8ccc31eb 207 push_lang_context (lang_name_c);
8d2733ca 208
9a0d1e1b
AM
209 set_exception_lang_code (EH_LANG_C_plus_plus);
210 set_exception_version_code (1);
9a0d1e1b 211
fb98cff6 212 CatchMatch
beb53fb8
JM
213 = builtin_function (flag_rtti
214 ? "__throw_type_match_rtti"
215 : "__throw_type_match",
216 build_function_type (ptr_type_node,
6c20b7e9
JM
217 tree_cons (NULL_TREE, const_ptr_type_node,
218 tree_cons (NULL_TREE, const_ptr_type_node,
beb53fb8
JM
219 tree_cons (NULL_TREE, ptr_type_node,
220 void_list_node)))),
221 NOT_BUILT_IN, NULL_PTR);
fb98cff6 222 FirstExceptionMatch
beb53fb8
JM
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);
fb98cff6 228 Unwind
beb53fb8
JM
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);
8d2733ca 234
8d2733ca 235 pop_lang_context ();
f30432d7 236
eb66be0e
MS
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;
8d2733ca
MS
240}
241
95e8dcba 242/* Retrieve a pointer to the cp_eh_info node for the current exception. */
6874c264 243
95e8dcba
JM
244static tree
245call_eh_info ()
6874c264 246{
95e8dcba 247 tree fn;
6874c264 248
a1622f83 249 fn = get_identifier ("__cp_eh_info");
6874c264
JM
250 if (IDENTIFIER_GLOBAL_VALUE (fn))
251 fn = IDENTIFIER_GLOBAL_VALUE (fn);
252 else
253 {
a1622f83 254 tree t1, t, fields[7];
6874c264 255
a1622f83 256 /* Declare cp_eh_info * __cp_eh_info (void),
6874c264
JM
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. */
9a0d1e1b
AM
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,
9a0d1e1b 279 get_identifier ("language"), short_integer_type_node);
5816cb14 280 fields[2] = build_lang_field_decl (FIELD_DECL,
9a0d1e1b
AM
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. */
5816cb14 284 finish_builtin_type (t1, "__eh_info", fields, 2, ptr_type_node);
6874c264 285 t = make_lang_type (RECORD_TYPE);
a1622f83
AM
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"),
6874c264 289 ptr_type_node);
a1622f83 290 fields[2] = build_lang_field_decl (FIELD_DECL, get_identifier ("type"),
6874c264 291 ptr_type_node);
a1622f83 292 fields[3] = build_lang_field_decl
6874c264
JM
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))));
a1622f83 297 fields[4] = build_lang_field_decl (FIELD_DECL, get_identifier ("caught"),
6874c264 298 boolean_type_node);
a1622f83 299 fields[5] = build_lang_field_decl (FIELD_DECL, get_identifier ("next"),
6874c264 300 build_pointer_type (t));
a1622f83 301 fields[6] = build_lang_field_decl
20b90169 302 (FIELD_DECL, get_identifier ("handlers"), long_integer_type_node);
cf9d67e3
BK
303 /* N.B.: The fourth field LEN is expected to be
304 the number of fields - 1, not the total number of fields. */
a1622f83 305 finish_builtin_type (t, "cp_eh_info", fields, 6, ptr_type_node);
6874c264
JM
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 }
95e8dcba
JM
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
325static void
326push_eh_info ()
327{
328 tree decl, fn = call_eh_info ();
6874c264
JM
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
342static tree
343get_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
352static tree
353get_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
361static tree
362get_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
371static tree
372get_eh_caught ()
373{
374 return build_component_ref (get_eh_info (), get_identifier ("caught"),
375 NULL_TREE, 0);
376}
377
20b90169
JM
378/* Returns a reference to whether or not the current exception
379 has been caught. */
380
381static tree
382get_eh_handlers ()
383{
384 return build_component_ref (get_eh_info (), get_identifier ("handlers"),
385 NULL_TREE, 0);
386}
387
f30432d7
MS
388/* Build a type value for use at runtime for a type that is matched
389 against by the exception handling system. */
6467930b 390
f30432d7
MS
391static tree
392build_eh_type_type (type)
393 tree type;
8d08fdba 394{
f30432d7
MS
395 char *typestring;
396 tree exp;
8d2733ca 397
f30432d7
MS
398 if (type == error_mark_node)
399 return error_mark_node;
8d2733ca 400
e92cc029 401 /* peel back references, so they match. */
f30432d7
MS
402 if (TREE_CODE (type) == REFERENCE_TYPE)
403 type = TREE_TYPE (type);
8d08fdba 404
e92cc029 405 /* Peel off cv qualifiers. */
f30432d7 406 type = TYPE_MAIN_VARIANT (type);
8d2733ca 407
f30432d7 408 if (flag_rtti)
8d08fdba 409 {
f30432d7 410 return build1 (ADDR_EXPR, ptr_type_node, get_typeid (type));
8d08fdba 411 }
f30432d7
MS
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);
8d08fdba 416}
8d08fdba 417
5816cb14
AM
418/* Build the address of a runtime type for use in the runtime matching
419 field of the new exception model */
420
421static tree
422build_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
faae18ab
MS
460/* Build a type value for use at runtime for a exp that is thrown or
461 matched against by the exception handling system. */
6467930b 462
faae18ab
MS
463static tree
464build_eh_type (exp)
465 tree exp;
466{
faae18ab
MS
467 if (flag_rtti)
468 {
469 exp = build_typeid (exp);
470 return build1 (ADDR_EXPR, ptr_type_node, exp);
471 }
f30432d7 472 return build_eh_type_type (TREE_TYPE (exp));
faae18ab
MS
473}
474
9c606f69
AM
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. */
478void
479mark_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
c7ae64f2
JM
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. */
6467930b 507
c7ae64f2 508static tree
de35891e 509do_pop_exception ()
72b7eeff 510{
6874c264 511 tree fn, cleanup;
6874c264
JM
512 fn = get_identifier ("__cp_pop_exception");
513 if (IDENTIFIER_GLOBAL_VALUE (fn))
514 fn = IDENTIFIER_GLOBAL_VALUE (fn);
515 else
516 {
c7ae64f2
JM
517 /* Declare void __cp_pop_exception (void *),
518 as defined in exception.cc. */
6874c264
JM
519 push_obstacks_nochange ();
520 end_temporary_allocation ();
c7ae64f2
JM
521 fn = build_lang_decl
522 (FUNCTION_DECL, fn,
523 build_function_type (void_type_node, tree_cons
de35891e 524 (NULL_TREE, ptr_type_node, void_list_node)));
6874c264
JM
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 }
72b7eeff
MS
533
534 /* Arrange to do a dynamically scoped cleanup upon exit from this region. */
c7ae64f2
JM
535 cleanup = lookup_name (get_identifier ("__exception_info"), 0);
536 cleanup = build_function_call (fn, expr_tree_cons
de35891e 537 (NULL_TREE, cleanup, NULL_TREE));
c0700ea5 538 return cleanup;
c7ae64f2
JM
539}
540
541/* This routine creates the cleanup for the current exception. */
72b7eeff 542
c7ae64f2
JM
543static void
544push_eh_cleanup ()
545{
de35891e 546 int yes;
72b7eeff 547
20b90169
JM
548 expand_expr (build_unary_op (PREINCREMENT_EXPR, get_eh_handlers (), 1),
549 const0_rtx, VOIDmode, EXPAND_NORMAL);
550
de35891e
JM
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);
c7ae64f2 555}
72b7eeff 556
f4a23343
JM
557/* Build up a call to terminate on the function obstack, for use as an
558 exception handler. */
559
560tree
561build_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
5816cb14 569/* Call this to start a catch block. Typename is the typename, and identifier
8d2733ca
MS
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) */
6467930b 574
8d08fdba 575void
a4443a08
MS
576expand_start_catch_block (declspecs, declarator)
577 tree declspecs, declarator;
8d08fdba 578{
5816cb14 579 tree decl;
8d2733ca 580
faf5394a
MS
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
8d2733ca
MS
596 if (! doing_eh (1))
597 return;
598
5816cb14
AM
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
610static void
611process_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
c7ae64f2
JM
618 /* Create a binding level for the eh_info and the exception object
619 cleanup. */
c11b6f21 620 pushlevel (0);
a3b49ccd
MS
621 expand_start_bindings (0);
622
f675499c 623 false_label_rtx = gen_label_rtx ();
72b7eeff 624 push_label_entry (&false_label_stack, false_label_rtx, NULL_TREE);
a3b49ccd 625
eb66be0e
MS
626 emit_line_note (input_filename, lineno);
627
6874c264
JM
628 push_eh_info ();
629
faae18ab 630 if (declspecs)
8d08fdba 631 {
c11b6f21 632 decl = grokdeclarator (declarator, declspecs, CATCHPARM, 1, NULL_TREE);
faae18ab
MS
633
634 if (decl == NULL_TREE)
c7ae64f2
JM
635 error ("invalid catch parameter");
636 }
95e8dcba 637
c7ae64f2
JM
638 if (decl)
639 {
640 tree exp;
641 rtx call_rtx, return_value_rtx;
642 tree init_type;
faae18ab 643
be99da77
MS
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
e92cc029 648 /* Figure out the type that the initializer is. */
faae18ab 649 init_type = TREE_TYPE (decl);
f30432d7
MS
650 if (TREE_CODE (init_type) != REFERENCE_TYPE
651 && TREE_CODE (init_type) != POINTER_TYPE)
faae18ab
MS
652 init_type = build_reference_type (init_type);
653
6874c264 654 exp = get_eh_value ();
92f5c135
JM
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
e66d884e 662 exp = expr_tree_cons (NULL_TREE,
f30432d7 663 build_eh_type_type (TREE_TYPE (decl)),
e66d884e 664 expr_tree_cons (NULL_TREE,
6874c264 665 get_eh_type (),
e66d884e 666 expr_tree_cons (NULL_TREE, exp, NULL_TREE)));
faae18ab
MS
667 exp = build_function_call (CatchMatch, exp);
668 call_rtx = expand_call (exp, NULL_RTX, 0);
8d2733ca 669
faae18ab 670 return_value_rtx = hard_function_value (ptr_type_node, exp);
8d2733ca
MS
671
672 /* did the throw type match function return TRUE? */
faae18ab 673 emit_cmp_insn (return_value_rtx, const0_rtx, EQ, NULL_RTX,
8d2733ca
MS
674 GET_MODE (return_value_rtx), 0, 0);
675
676 /* if it returned FALSE, jump over the catch block, else fall into it */
faae18ab
MS
677 emit_jump_insn (gen_beq (false_label_rtx));
678
95e8dcba
JM
679 push_eh_cleanup ();
680
c7ae64f2
JM
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);
f4a23343
JM
695 init = build (TRY_CATCH_EXPR, TREE_TYPE (init), init,
696 build_terminate_handler ());
c7ae64f2 697 }
faae18ab 698
b3417a04 699 /* Let `cp_finish_decl' know that this initializer is ok. */
faae18ab
MS
700 DECL_INITIAL (decl) = init;
701 decl = pushdecl (decl);
c7ae64f2 702
b3417a04 703 cp_finish_decl (decl, init, NULL_TREE, 0, LOOKUP_ONLYCONVERTING);
8d08fdba 704 }
95e8dcba
JM
705 else
706 {
707 push_eh_cleanup ();
708
c7ae64f2
JM
709 /* Create a binding level for the parm. */
710 pushlevel (0);
711 expand_start_bindings (0);
712
95e8dcba
JM
713 /* Fall into the catch all section. */
714 }
72b7eeff 715
6874c264
JM
716 init = build_modify_expr (get_eh_caught (), NOP_EXPR, integer_one_node);
717 expand_expr (init, const0_rtx, VOIDmode, EXPAND_NORMAL);
f30432d7 718
6467930b
MS
719 emit_line_note (input_filename, lineno);
720}
f30432d7 721
5816cb14
AM
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
727static void
728process_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
9c606f69 752 start_catch_handler (CATCH_ALL_TYPE);
5816cb14
AM
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
f30432d7 827
f30432d7 828
8d2733ca
MS
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. */
6467930b 832
824b9a4c
MS
833void
834expand_end_catch_block ()
8d08fdba 835{
f30432d7
MS
836 if (! doing_eh (1))
837 return;
8d2733ca 838
c7ae64f2
JM
839 /* Cleanup the EH parameter. */
840 expand_end_bindings (getdecls (), kept_level_p (), 0);
841 poplevel (kept_level_p (), 1, 0);
842
c7ae64f2
JM
843 /* Cleanup the EH object. */
844 expand_end_bindings (getdecls (), kept_level_p (), 0);
845 poplevel (kept_level_p (), 1, 0);
eb66be0e 846
eb66be0e
MS
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
6467930b 852 /* label we emit to jump to if this catch block didn't match. */
f30432d7 853 /* This the closing } in the `if (eq) {' of the documentation. */
5816cb14
AM
854 if (! flag_new_exceptions)
855 emit_label (pop_label_entry (&false_label_stack));
8d2733ca 856}
8d08fdba 857
6c20b7e9
JM
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. */
8d2733ca 868
f30432d7
MS
869void
870expand_start_eh_spec ()
871{
6c20b7e9 872 expand_start_try_stmts ();
f30432d7
MS
873}
874
5566b478 875static void
f30432d7
MS
876expand_end_eh_spec (raises)
877 tree raises;
878{
6c20b7e9
JM
879 tree tmp, fn, decl, types = NULL_TREE;
880 int count = 0;
f30432d7 881
6c20b7e9
JM
882 expand_start_all_catch ();
883 expand_start_catch_block (NULL_TREE, NULL_TREE);
f30432d7 884
6c20b7e9
JM
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 }
eb66be0e 892
6c20b7e9
JM
893 types = build_nt (CONSTRUCTOR, NULL_TREE, types);
894 TREE_HAS_CONSTRUCTOR (types) = 1;
6874c264 895
6c20b7e9
JM
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);
6874c264 904
6c20b7e9
JM
905 fn = get_identifier ("__check_eh_spec");
906 if (IDENTIFIER_GLOBAL_VALUE (fn))
907 fn = IDENTIFIER_GLOBAL_VALUE (fn);
908 else
f30432d7 909 {
6c20b7e9
JM
910 push_obstacks_nochange ();
911 end_temporary_allocation ();
f30432d7 912
6c20b7e9
JM
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);
f30432d7 917
6c20b7e9
JM
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 ();
f30432d7
MS
936}
937
8d2733ca
MS
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. */
6467930b 941
8d08fdba 942void
8d2733ca 943expand_exception_blocks ()
8d08fdba 944{
92b96838 945 do_pending_stack_adjust ();
e00737d2 946 push_to_sequence (catch_clauses);
8d2733ca 947 expand_leftover_cleanups ();
92b96838 948 do_pending_stack_adjust ();
e00737d2 949 catch_clauses = get_insns ();
f30432d7 950 end_sequence ();
eb448459 951
eb448459
MS
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. */
f30432d7
MS
956 if (TYPE_RAISES_EXCEPTIONS (TREE_TYPE (current_function_decl)))
957 {
eb448459 958 expand_end_eh_spec (TYPE_RAISES_EXCEPTIONS (TREE_TYPE (current_function_decl)));
92b96838 959 do_pending_stack_adjust ();
e00737d2 960 push_to_sequence (catch_clauses);
eb448459 961 expand_leftover_cleanups ();
92b96838 962 do_pending_stack_adjust ();
e00737d2 963 catch_clauses = get_insns ();
eb448459 964 end_sequence ();
f30432d7
MS
965 }
966
e00737d2 967 if (catch_clauses)
f30432d7 968 {
e00737d2
MS
969 rtx funcend = gen_label_rtx ();
970 emit_jump (funcend);
971
eb66be0e
MS
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)
fb98cff6 976 expand_eh_region_start ();
f30432d7 977
e00737d2
MS
978 emit_insns (catch_clauses);
979 catch_clauses = NULL_RTX;
eb66be0e
MS
980
981 if (exceptions_via_longjmp == 0)
f4a23343 982 expand_eh_region_end (build_terminate_handler ());
eb66be0e 983
6467930b 984 expand_leftover_cleanups ();
f30432d7 985
e00737d2
MS
986 emit_label (funcend);
987 }
8d08fdba
MS
988}
989
72b7eeff
MS
990tree
991start_anon_func ()
992{
993 static int counter = 0;
e92cc029 994 int old_interface_unknown = interface_unknown;
72b7eeff
MS
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
e92cc029
MS
1005 interface_unknown = 1;
1006
72b7eeff 1007 params = void_list_node;
956d6950 1008 /* tcf stands for throw clean function. */
72b7eeff 1009 sprintf (name, "__tcf_%d", counter++);
c11b6f21
MS
1010 t = make_call_declarator (get_identifier (name), params, NULL_TREE,
1011 NULL_TREE);
72b7eeff
MS
1012 start_function (decl_tree_cons (NULL_TREE, get_identifier ("static"),
1013 void_list_node),
c11b6f21 1014 t, NULL_TREE, 0);
72b7eeff
MS
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
e92cc029
MS
1022 interface_unknown = old_interface_unknown;
1023
72b7eeff
MS
1024 pop_lang_context ();
1025
1026 return current_function_decl;
1027}
1028
1029void
1030end_anon_func ()
1031{
fc378698 1032 expand_end_bindings (getdecls (), 1, 0);
72b7eeff
MS
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}
8d2733ca 1041
f4a23343
JM
1042/* Return a pointer to a buffer for an exception object of type TYPE. */
1043
1044tree
1045alloc_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
6467930b 1077/* Expand a throw statement. This follows the following
8d2733ca
MS
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
e1cd6e56 1082 newly allocated stack space since we can't save the pc directly.
8d2733ca
MS
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. */
6467930b 1086
8d08fdba 1087void
8d2733ca
MS
1088expand_throw (exp)
1089 tree exp;
8d08fdba 1090{
6874c264
JM
1091 tree fn;
1092 static tree cleanup_type;
8d08fdba 1093
8d2733ca
MS
1094 if (! doing_eh (1))
1095 return;
8d08fdba 1096
8d2733ca
MS
1097 if (exp)
1098 {
faae18ab 1099 tree throw_type;
6874c264 1100 tree cleanup = NULL_TREE, e;
faae18ab 1101
a3b49ccd 1102 /* throw expression */
e92cc029 1103 /* First, decay it. */
f30432d7 1104 exp = decay_conversion (exp);
a3b49ccd 1105
6874c264
JM
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
f30432d7
MS
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 {
f4a23343
JM
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. */
a50f0918 1142
f4a23343
JM
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);
faae18ab 1174
f30432d7
MS
1175 if (exp == error_mark_node)
1176 error (" in thrown expression");
faae18ab 1177
f4a23343
JM
1178 expand_expr (exp, const0_rtx, VOIDmode, 0);
1179 expand_eh_region_end (build_terminate_handler ());
1180 expand_end_target_temps ();
1181
72b7eeff
MS
1182 throw_type = build_eh_type (object);
1183
6874c264
JM
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);
e872bb7a 1189 mark_used (cleanup);
6874c264
JM
1190 mark_addressable (cleanup);
1191 /* Pretend it's a normal function. */
1192 cleanup = build1 (ADDR_EXPR, cleanup_type, cleanup);
1193 }
f4a23343
JM
1194
1195 exp = ptr;
f30432d7 1196 }
faae18ab 1197
6874c264
JM
1198 if (cleanup == NULL_TREE)
1199 {
1200 cleanup = build_int_2 (0, 0);
1201 TREE_TYPE (cleanup) = cleanup_type;
1202 }
72b7eeff 1203
6874c264
JM
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 }
72b7eeff 1228
6874c264
JM
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);
8d2733ca
MS
1234 }
1235 else
a3b49ccd 1236 {
6874c264
JM
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);
a3b49ccd 1262 }
8d2733ca 1263
e701eb4d 1264 expand_internal_throw ();
f376e137 1265}
8d2733ca
MS
1266
1267/* Build a throw expression. */
6467930b 1268
8d2733ca
MS
1269tree
1270build_throw (e)
1271 tree e;
1272{
02020185
JM
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)
db5ae43f 1280 {
02020185
JM
1281 cp_warning ("throwing NULL");
1282 e = integer_zero_node;
db5ae43f 1283 }
02020185
JM
1284
1285 e = build1 (THROW_EXPR, void_type_node, e);
1286 TREE_SIDE_EFFECTS (e) = 1;
1287 TREE_USED (e) = 1;
1288
8d2733ca 1289 return e;
8d08fdba 1290}
This page took 0.457343 seconds and 5 git commands to generate.