// Copyright (C) 2012 The Android Open Source Project // All rights reserved. // // Redistribution and use in source and binary forms, with or without // modification, are permitted provided that the following conditions // are met: // 1. Redistributions of source code must retain the above copyright // notice, this list of conditions and the following disclaimer. // 2. Redistributions in binary form must reproduce the above copyright // notice, this list of conditions and the following disclaimer in the // documentation and/or other materials provided with the distribution. // 3. Neither the name of the project nor the names of its contributors // may be used to endorse or promote products derived from this software // without specific prior written permission. // // THIS SOFTWARE IS PROVIDED BY THE PROJECT AND CONTRIBUTORS ``AS IS'' AND // ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE // IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE // ARE DISCLAIMED. IN NO EVENT SHALL THE PROJECT OR CONTRIBUTORS BE LIABLE // FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL // DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS // OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) // HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT // LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY // OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF // SUCH DAMAGE. //===----------------------------------------------------------------------===// // The LLVM Compiler Infrastructure // // This file is dual licensed under the MIT and the University of Illinois Open // Source Licenses. See LICENSE.TXT for details. // // // This file implements the "Exception Handling APIs" // http://www.codesourcery.com/public/cxx-abi/abi-eh.html // http://www.intel.com/design/itanium/downloads/245358.htm // //===----------------------------------------------------------------------===// /* * Copyright 2010-2011 PathScale, Inc. All rights reserved. * * Redistribution and use in source and binary forms, with or without * modification, are permitted provided that the following conditions are met: * * 1. Redistributions of source code must retain the above copyright notice, * this list of conditions and the following disclaimer. * * 2. Redistributions in binary form must reproduce the above copyright notice, * this list of conditions and the following disclaimer in the documentation * and/or other materials provided with the distribution. * * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS ``AS * IS'' AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, * THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR * PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR * CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, * EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, * PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; * OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, * WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR * OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF * ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. */ #include <cstdlib> #include <cxxabi.h> #include <unwind.h> #include "dwarf_helper.h" #include "helper_func_internal.h" namespace __cxxabiv1 { #ifdef __arm__ extern "C" enum type_match_result { ctm_failed = 0, ctm_succeeded = 1, ctm_succeeded_with_ptr_to_base = 2 }; extern "C" type_match_result __attribute__((visibility("default"))) __cxa_type_match(_Unwind_Exception* ucbp, const __shim_type_info* rttip, bool is_reference_type, void** matched_object) { __cxa_exception* header = reinterpret_cast<__cxa_exception*>(ucbp+1)-1; type_match_result result = ctm_succeeded; void* adjustedPtr = header+1; if (dynamic_cast<const __pointer_type_info*>(header->exceptionType)) { adjustedPtr = *reinterpret_cast<void**>(adjustedPtr); result = ctm_succeeded_with_ptr_to_base; } const __shim_type_info* catch_type = rttip; const __shim_type_info* thrown_type = static_cast<const __shim_type_info*>(header->exceptionType); if (!catch_type || !thrown_type) { return ctm_failed; } if (catch_type->can_catch(thrown_type, adjustedPtr)) { *matched_object = adjustedPtr; return result; } return ctm_failed; } #endif // __arm__ namespace { void terminate_helper(std::terminate_handler t_handler) { try { t_handler(); abort(); } catch (...) { abort(); } } void unexpected_helper(std::unexpected_handler u_handler) { u_handler(); std::terminate(); } } // namespace #ifdef __arm__ extern "C" bool __attribute__((visibility("default"))) __cxa_begin_cleanup(_Unwind_Exception* exc) { __cxa_eh_globals *globals = __cxa_get_globals(); __cxa_exception *header = reinterpret_cast<__cxa_exception*>(exc+1)-1; bool native = header->unwindHeader.exception_class == __gxx_exception_class; if (native) { header->cleanupCount += 1; if (header->cleanupCount == 1) { // First time header->nextCleanup = globals->cleanupExceptions; globals->cleanupExceptions = header; } } else { globals->cleanupExceptions = header; } return true; } extern "C" _Unwind_Exception * helper_end_cleanup() { __cxa_eh_globals *globals = __cxa_get_globals(); __cxa_exception* header = globals->cleanupExceptions; if (!header) { std::terminate(); } if (header->unwindHeader.exception_class == __gxx_exception_class) { header->cleanupCount -= 1; if (header->cleanupCount == 0) { // Last one globals->cleanupExceptions = header->nextCleanup; header->nextCleanup = NULL; } } else { globals->cleanupExceptions = NULL; } return &header->unwindHeader; } asm ( ".pushsection .text.__cxa_end_cleanup \n" ".global __cxa_end_cleanup \n" ".type __cxa_end_cleanup, \"function\" \n" "__cxa_end_cleanup: \n" " push\t{r1, r2, r3, r4} \n" " bl helper_end_cleanup \n" " pop\t{r1, r2, r3, r4} \n" " bl _Unwind_Resume \n" " bl abort \n" ".popsection \n" ); extern "C" void __attribute__((visibility("default"))) __cxa_call_unexpected(void* arg) { _Unwind_Exception* unwind_exception = static_cast<_Unwind_Exception*>(arg); __cxa_exception* header = reinterpret_cast<__cxa_exception*>(unwind_exception+1)-1; bool native_exception = unwind_exception->exception_class == __gxx_exception_class; if (!native_exception) { __cxa_begin_catch(unwind_exception); // unexpected is also a handler try { std::unexpected(); } catch (...) { std::terminate(); } return; } // Cache previous data first since we will change contents below. uint32_t count = unwind_exception->barrier_cache.bitpattern[1]; uint32_t stride = unwind_exception->barrier_cache.bitpattern[3]; uint32_t* list = reinterpret_cast<uint32_t*>( unwind_exception->barrier_cache.bitpattern[4]); __cxa_begin_catch(unwind_exception); // unexpected is also a handler try { unexpected_helper(header->unexpectedHandler); } catch (...) { // A new exception thrown when calling unexpected. bool allow_bad_exception = false; for (uint32_t i = 0; i != count; ++i) { uint32_t offset = reinterpret_cast<uint32_t>(&list[i * (stride >> 2)]); offset = decodeRelocTarget2(offset); const __shim_type_info* catch_type = reinterpret_cast<const __shim_type_info*>(offset); __cxa_exception* new_header = __cxa_get_globals()->caughtExceptions; void* adjustedPtr = new_header + 1; if (__cxa_type_match(&new_header->unwindHeader, catch_type, false/* is_ref_type */, &adjustedPtr) != ctm_failed) { throw; } void* null_adjustedPtr = NULL; const __shim_type_info* bad_excp = static_cast<const __shim_type_info*>(&typeid(std::bad_exception)); if (catch_type->can_catch(bad_excp, null_adjustedPtr)) { allow_bad_exception = true; } } // If no other ones match, throw bad_exception. if (allow_bad_exception) { __cxa_end_catch(); __cxa_end_catch(); throw std::bad_exception(); } terminate_helper(header->terminateHandler); } } #else // ! __arm__ extern "C" void __attribute__((visibility("default"))) __cxa_call_unexpected(void* arg) { _Unwind_Exception* unwind_exception = static_cast<_Unwind_Exception*>(arg); if (unwind_exception == 0) { call_terminate(unwind_exception); } __cxa_begin_catch(unwind_exception); // unexpected is also a handler bool native_old_exception = unwind_exception->exception_class == __gxx_exception_class; std::unexpected_handler u_handler; std::terminate_handler t_handler; __cxa_exception* old_exception_header = 0; int64_t ttypeIndex; const uint8_t* lsda; if (native_old_exception) { old_exception_header = reinterpret_cast<__cxa_exception*>(unwind_exception+1)-1; t_handler = old_exception_header->terminateHandler; u_handler = old_exception_header->unexpectedHandler; // If unexpected_helper(u_handler) rethrows the same exception, // these values get overwritten by the rethrow. So save them now: ttypeIndex = old_exception_header->handlerSwitchValue; lsda = old_exception_header->languageSpecificData; } else { t_handler = std::get_terminate(); u_handler = std::get_unexpected(); } try { unexpected_helper(u_handler); } catch (...) { // A new exception thrown when calling unexpected. if (!native_old_exception) { std::terminate(); } uint8_t lpStartEncoding = *lsda++; const uint8_t* lpStart = (const uint8_t*)readEncodedPointer(&lsda, lpStartEncoding); uint8_t ttypeEncoding = *lsda++; if (ttypeEncoding == DW_EH_PE_omit) { terminate_helper(t_handler); } uintptr_t classInfoOffset = readULEB128(&lsda); const uint8_t* classInfo = lsda + classInfoOffset; __cxa_eh_globals* globals = __cxa_get_globals_fast(); __cxa_exception* new_exception_header = globals->caughtExceptions; if (new_exception_header == 0) { // This shouldn't be able to happen! terminate_helper(t_handler); } bool native_new_exception = new_exception_header->unwindHeader.exception_class == __gxx_exception_class; if (native_new_exception && (new_exception_header != old_exception_header)) { const std::type_info* excpType = new_exception_header->exceptionType; if (!canExceptionSpecCatch(ttypeIndex, classInfo, ttypeEncoding, excpType, new_exception_header+1, unwind_exception)) { // We need to __cxa_end_catch, but for the old exception, // not the new one. This is a little tricky ... // Disguise new_exception_header as a rethrown exception, but // don't actually rethrow it. This means you can temporarily // end the catch clause enclosing new_exception_header without // __cxa_end_catch destroying new_exception_header. new_exception_header->handlerCount = -new_exception_header->handlerCount; globals->uncaughtExceptions += 1; __cxa_end_catch(); __cxa_end_catch(); __cxa_begin_catch(&new_exception_header->unwindHeader); throw; } } const std::type_info* excpType = &typeid(std::bad_exception); if (!canExceptionSpecCatch(ttypeIndex, classInfo, ttypeEncoding, excpType, NULL, unwind_exception)) { __cxa_end_catch(); __cxa_end_catch(); throw std::bad_exception(); } } // catch (...) // Call terminate after unexpected normally done terminate_helper(t_handler); } #endif // __arm__ } // namespace __cxxabiv1