// -*- C++ -*- Exception handling routines for catching. // Copyright (C) 2001-2015 Free Software Foundation, Inc. // // This file is part of GCC. // // GCC 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 3, or (at your option) // any later version. // // GCC 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. // // Under Section 7 of GPL version 3, you are granted additional // permissions described in the GCC Runtime Library Exception, version // 3.1, as published by the Free Software Foundation. // You should have received a copy of the GNU General Public License and // a copy of the GCC Runtime Library Exception along with this program; // see the files COPYING3 and COPYING.RUNTIME respectively. If not, see // . #include #include "unwind-cxx.h" using namespace __cxxabiv1; extern "C" void * __cxxabiv1::__cxa_get_exception_ptr(void *exc_obj_in) _GLIBCXX_NOTHROW { _Unwind_Exception *exceptionObject = reinterpret_cast <_Unwind_Exception *>(exc_obj_in); return __gxx_caught_object(exceptionObject); } extern "C" void * __cxxabiv1::__cxa_begin_catch (void *exc_obj_in) _GLIBCXX_NOTHROW { _Unwind_Exception *exceptionObject = reinterpret_cast <_Unwind_Exception *>(exc_obj_in); __cxa_eh_globals *globals = __cxa_get_globals (); __cxa_exception *prev = globals->caughtExceptions; __cxa_exception *header = __get_exception_header_from_ue (exceptionObject); void* objectp; // Foreign exceptions can't be stacked here. If the exception stack is // empty, then fine. Otherwise we really have no choice but to terminate. // Note that this use of "header" is a lie. It's fine so long as we only // examine header->unwindHeader though. if (!__is_gxx_exception_class(header->unwindHeader.exception_class)) { if (prev != 0) std::terminate (); // Remember for end_catch and rethrow. globals->caughtExceptions = header; // ??? No sensible value to return; we don't know what the // object is, much less where it is in relation to the header. return 0; } int count = header->handlerCount; // Count is less than zero if this exception was rethrown from an // immediately enclosing region. if (count < 0) count = -count + 1; else count += 1; header->handlerCount = count; globals->uncaughtExceptions -= 1; if (header != prev) { header->nextException = prev; globals->caughtExceptions = header; } objectp = __gxx_caught_object(exceptionObject); PROBE2 (catch, objectp, header->exceptionType); #ifdef __ARM_EABI_UNWINDER__ _Unwind_Complete(exceptionObject); #endif return objectp; } extern "C" void __cxxabiv1::__cxa_end_catch () { __cxa_eh_globals *globals = __cxa_get_globals_fast (); __cxa_exception *header = globals->caughtExceptions; // A rethrow of a foreign exception will be removed from the // the exception stack immediately by __cxa_rethrow. if (!header) return; // A foreign exception couldn't have been stacked (see above), // so by definition processing must be complete. if (!__is_gxx_exception_class(header->unwindHeader.exception_class)) { globals->caughtExceptions = 0; _Unwind_DeleteException (&header->unwindHeader); return; } int count = header->handlerCount; if (count < 0) { // This exception was rethrown. Decrement the (inverted) catch // count and remove it from the chain when it reaches zero. if (++count == 0) globals->caughtExceptions = header->nextException; } else if (--count == 0) { // Handling for this exception is complete. Destroy the object. globals->caughtExceptions = header->nextException; _Unwind_DeleteException (&header->unwindHeader); return; } else if (count < 0) // A bug in the exception handling library or compiler. std::terminate (); header->handlerCount = count; } bool std::uncaught_exception() throw() { #if __cpp_exceptions __cxa_eh_globals *globals = __cxa_get_globals (); return globals->uncaughtExceptions != 0; #else return false; #endif } int std::uncaught_exceptions() throw() { #if __cpp_exceptions __cxa_eh_globals *globals = __cxa_get_globals (); return globals->uncaughtExceptions; #else return 0; #endif }