154bba13a4
Thu Dec 11 20:42:18 1997 Teemu Torma <tot@trema.com> Thread-safe EH support for pthreads, DCE threads and Solaris threads. * integrate.c (expand_inline_function): If the inline fn uses eh context, make sure that the current fn has one. * toplev.c (rest_of_compilation): Call emit_eh_context. * except.c (use_eh_context): New fn. (get_eh_context_once): New fn. (call_get_eh_context): New fn. (emit_eh_context): New fn. (get_eh_context): Call either get_eh_context_once or call_get_eh_context, depending on what we have. (get_dynamic_handler_chain): Call get_eh_context_once. * except.h: Prototypes for fns above. * optabs.c (get_eh_context_libfunc): Removed. (init_optabs): Don't initialize it. * expr.h (get_eh_context_libfunc): Removed. * rtl.h, rtl.c: New reg_note REG_EH_CONTEXT. * config/pa/pa.h (CPP_SPEC): Support for -threads. * config/pa/pa-hpux10.h (LIB_SPEC): Ditto. * config/pa/t-pa (MULTILIB_OPTIONS, MULTILIB_DIRNAMES): New multilib for -threads. * config/sparc/t-sol2: Added multilibs for -threads and made -pthreads alias to it. * config/sparc/sol2.h (CPP_SPEC, LIB_SPEC): Added -threads and -pthreads options. * libgcc-thr.h: New file. * libgcc2.c: (__get_cpp_eh_context): Removed. (struct cpp_eh_context): Removed. (struct eh_context): Replaced cpp_eh_context with generic language specific pointer. (__get_eh_info): New function. (__throw): Check eh_context::info. (__sjthrow): Ditto. * libgcc2.c: Include libgcc-thr.h. (new_eh_context, __get_eh_context, eh_pthread_initialize, eh_context_initialize, eh_context_static, eh_context_specific, eh_context_free): New functions. (get_eh_context, eh_context_key): New variables. (__sjthrow, __sjpopnthrow, __eh_pcnthrow, __throw): Use get_eh_context to get the context. (longjmp): Move the declaration inside #ifdef DONT_USE_BUILTIN_SETJMP. * frame.c: Include libgcc-thr.h. (object_mutex): Mutex to protect the object list. (find_fde, __register_frame, __register_frame_table, __deregister_frame): Hold the lock while accessing objects. * except.h (get_eh_context): Declare. * except.c (current_function_ehc): Define. (current_function_dhc, current_function_dcc): Removed. (get_eh_context): New function. (get_dynamic_handler_chain): Use get_eh_context. (get_saved_pc_ref): Ditto. (get_dynamic_cleanup_chain): Removed references to current_function_dcc. (save_eh_status, restore_eh_status): Save and restore current_function_ehc instead. * optabs.c (get_eh_context_libfunc): New variable. (init_optabs): Initialize it. * expr.h: Declare get_eh_context_libfunc. * function.h (struct function): Replaced dhc and dcc with ehc. * except.c (get_saved_pc_ref): New functions. (eh_saved_pc_rtx, eh_saved_pc): Deleted. (expand_internal_throw_indirect): Use get_saved_pc_ref() instead of eh_saved_pc. (end_eh_unwinder): Likewise. (init_eh): Remove initialization of eh_saved_pc. * optabs.c (get_saved_pc_libfunc): New variable. (init_optabs): Initialize it. * expr.h: Declare get_saved_pc_libfunc. * except.h (eh_saved_pc_rtx): Deleted. (get_saved_pc_ref): Declared. From Scott Snyder <snyder@d0sgif.fnal.gov>: * libgcc2.c (__get_saved_pc): New. (__eh_type, __eh_pc): Deleted. (__eh_pcnthrow): Use __get_saved_pc() instead of __eh_pc. (__get_dynamic_handler_chain): Move __dynamic_handler_chain inside this fcn. cp/: Thu Dec 11 20:43:33 1997 Teemu Torma <tot@trema.com> * decl.c (ptr_ptr_type_node): Define. (init_decl_processing): Initialize it. * cp-tree.h: Declare it. * exception.cc (__cp_exception_info): Use __get_eh_info. (__cp_push_exception): Ditto. (__cp_pop_exception): Ditto. From Scott Snyder <snyder@d0sgif.fnal.gov>: * except.c (expand_builtin_throw): Use get_saved_pc_ref instead of saved_pc. (init_exception_processing): Removed saved_pc initialization. From-SVN: r17052
254 lines
5.9 KiB
C++
254 lines
5.9 KiB
C++
// Functions for Exception Support for -*- C++ -*-
|
|
// Copyright (C) 1994, 1995, 1996 Free Software Foundation
|
|
|
|
// This file is part of GNU CC.
|
|
|
|
// GNU CC is free software; you can redistribute it and/or modify
|
|
// it under the terms of the GNU General Public License as published by
|
|
// the Free Software Foundation; either version 2, or (at your option)
|
|
// any later version.
|
|
|
|
// GNU CC is distributed in the hope that it will be useful,
|
|
// but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
// GNU General Public License for more details.
|
|
|
|
// You should have received a copy of the GNU General Public License
|
|
// along with GNU CC; see the file COPYING. If not, write to
|
|
// the Free Software Foundation, 59 Temple Place - Suite 330,
|
|
// Boston, MA 02111-1307, USA.
|
|
|
|
// As a special exception, if you link this library with other files,
|
|
// some of which are compiled with GCC, to produce an executable,
|
|
// this library does not by itself cause the resulting executable
|
|
// to be covered by the GNU General Public License.
|
|
// This exception does not however invalidate any other reasons why
|
|
// the executable file might be covered by the GNU General Public License.
|
|
|
|
#pragma implementation "exception"
|
|
|
|
#include "typeinfo"
|
|
#include "exception"
|
|
|
|
/* Define terminate, unexpected, set_terminate, set_unexpected as
|
|
well as the default terminate func and default unexpected func. */
|
|
|
|
extern terminate_handler __terminate_func __attribute__((__noreturn__));
|
|
|
|
void
|
|
terminate ()
|
|
{
|
|
__terminate_func ();
|
|
}
|
|
|
|
void
|
|
__default_unexpected ()
|
|
{
|
|
terminate ();
|
|
}
|
|
|
|
static unexpected_handler __unexpected_func __attribute__((__noreturn__))
|
|
= __default_unexpected;
|
|
|
|
terminate_handler
|
|
set_terminate (terminate_handler func)
|
|
{
|
|
terminate_handler old = __terminate_func;
|
|
|
|
__terminate_func = func;
|
|
return old;
|
|
}
|
|
|
|
unexpected_handler
|
|
set_unexpected (unexpected_handler func)
|
|
{
|
|
unexpected_handler old = __unexpected_func;
|
|
|
|
__unexpected_func = func;
|
|
return old;
|
|
}
|
|
|
|
void
|
|
unexpected ()
|
|
{
|
|
__unexpected_func ();
|
|
}
|
|
|
|
/* C++-specific state about the current exception.
|
|
This must match init_exception_processing().
|
|
|
|
Note that handlers and caught are not redundant; when rethrown, an
|
|
exception can have multiple active handlers and still be considered
|
|
uncaught. */
|
|
|
|
struct cp_eh_info
|
|
{
|
|
void *value;
|
|
void *type;
|
|
void (*cleanup)(void *, int);
|
|
bool caught;
|
|
cp_eh_info *next;
|
|
long handlers;
|
|
};
|
|
|
|
/* Language-specific EH info pointer, defined in libgcc2. */
|
|
|
|
extern "C" cp_eh_info **__get_eh_info (); // actually void **
|
|
|
|
/* Is P the type_info node for a pointer of some kind? */
|
|
|
|
extern bool __is_pointer (void *);
|
|
|
|
/* Compiler hook to return a pointer to the info for the current exception.
|
|
Used by get_eh_info (). */
|
|
|
|
extern "C" cp_eh_info *
|
|
__cp_exception_info (void)
|
|
{
|
|
return *__get_eh_info ();
|
|
}
|
|
|
|
/* Compiler hook to push a new exception onto the stack.
|
|
Used by expand_throw(). */
|
|
|
|
extern "C" void
|
|
__cp_push_exception (void *value, void *type, void (*cleanup)(void *, int))
|
|
{
|
|
cp_eh_info *p = new cp_eh_info;
|
|
p->value = value;
|
|
p->type = type;
|
|
p->cleanup = cleanup;
|
|
p->handlers = 0;
|
|
p->caught = false;
|
|
|
|
cp_eh_info **q = __get_eh_info ();
|
|
|
|
p->next = *q;
|
|
*q = p;
|
|
}
|
|
|
|
/* Compiler hook to pop an exception that has been finalized. Used by
|
|
push_eh_cleanup(). P is the info for the exception caught by the
|
|
current catch block. */
|
|
|
|
extern "C" void
|
|
__cp_pop_exception (cp_eh_info *p)
|
|
{
|
|
cp_eh_info **q = __get_eh_info ();
|
|
|
|
--p->handlers;
|
|
|
|
/* Don't really pop if there are still active handlers for our exception,
|
|
or if our exception is being rethrown (i.e. if the active exception is
|
|
our exception and it is uncaught). */
|
|
if (p->handlers != 0
|
|
|| (p == *q && !p->caught))
|
|
return;
|
|
|
|
for (; *q; q = &((*q)->next))
|
|
if (*q == p)
|
|
break;
|
|
|
|
if (! *q)
|
|
terminate ();
|
|
|
|
*q = p->next;
|
|
|
|
if (p->cleanup)
|
|
/* 3 is a magic value for destructors; see build_delete(). */
|
|
p->cleanup (p->value, 3);
|
|
else if (__is_pointer (p->type))
|
|
/* do nothing; pointers are passed directly in p->value. */;
|
|
else
|
|
delete p->value;
|
|
|
|
delete p;
|
|
}
|
|
|
|
extern "C" void
|
|
__uncatch_exception (void)
|
|
{
|
|
cp_eh_info *p = __cp_exception_info ();
|
|
if (p)
|
|
p->caught = false;
|
|
/* otherwise __throw will call terminate(); don't crash here. */
|
|
}
|
|
|
|
/* As per [except.unexpected]:
|
|
If an exception is thrown, we check it against the spec. If it doesn't
|
|
match, we call unexpected (). If unexpected () throws, we check that
|
|
exception against the spec. If it doesn't match, if the spec allows
|
|
bad_exception we throw that; otherwise we call terminate ().
|
|
|
|
The compiler treats an exception spec as a try block with a generic
|
|
handler that just calls this function with a list of the allowed
|
|
exception types, so we have an active exception that can be rethrown.
|
|
|
|
This function does not return. */
|
|
|
|
extern "C" void
|
|
__check_eh_spec (int n, const void **spec)
|
|
{
|
|
cp_eh_info *p = __cp_exception_info ();
|
|
|
|
for (int i = 0; i < n; ++i)
|
|
{
|
|
if (__throw_type_match_rtti (spec[i], p->type, p->value))
|
|
throw;
|
|
}
|
|
|
|
try
|
|
{
|
|
unexpected ();
|
|
}
|
|
catch (...)
|
|
{
|
|
// __exception_info is an artificial var pushed into each catch block.
|
|
if (p != __exception_info)
|
|
{
|
|
p = __exception_info;
|
|
for (int i = 0; i < n; ++i)
|
|
{
|
|
if (__throw_type_match_rtti (spec[i], p->type, p->value))
|
|
throw;
|
|
}
|
|
}
|
|
|
|
const type_info &bad_exc = typeid (bad_exception);
|
|
for (int i = 0; i < n; ++i)
|
|
{
|
|
if (__throw_type_match_rtti (spec[i], &bad_exc, p->value))
|
|
throw bad_exception ();
|
|
}
|
|
|
|
terminate ();
|
|
}
|
|
}
|
|
|
|
extern "C" void
|
|
__throw_bad_cast (void)
|
|
{
|
|
throw bad_cast ();
|
|
}
|
|
|
|
extern "C" void
|
|
__throw_bad_typeid (void)
|
|
{
|
|
throw bad_typeid ();
|
|
}
|
|
|
|
/* Has the current exception been caught? */
|
|
|
|
bool
|
|
uncaught_exception ()
|
|
{
|
|
cp_eh_info *p = __cp_exception_info ();
|
|
return p && ! p->caught;
|
|
}
|
|
|
|
const char * exception::
|
|
what () const
|
|
{
|
|
return typeid (*this).name ();
|
|
}
|