/* -*- Mode: C++; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4; fill-column: 100 -*- */ /* * This file is part of the LibreOffice project. * * This Source Code Form is subject to the terms of the Mozilla Public * License, v. 2.0. If a copy of the MPL was not distributed with this * file, You can obtain one at http://mozilla.org/MPL/2.0/. * * This file incorporates work covered by the following license notice: * * Licensed to the Apache Software Foundation (ASF) under one or more * contributor license agreements. See the NOTICE file distributed * with this work for additional information regarding copyright * ownership. The ASF licenses this file to you under the Apache * License, Version 2.0 (the "License"); you may not use this file * except in compliance with the License. You may obtain a copy of * the License at http://www.apache.org/licenses/LICENSE-2.0 . */ #include #include #include #include #include #include #include #include #include #include #include #include #include "share.hxx" using namespace ::std; using namespace ::osl; using namespace ::com::sun::star::uno; using namespace ::__cxxabiv1; //#define BRIDGE_DEBUG namespace CPPU_CURRENT_NAMESPACE { void dummy_can_throw_anything(char const*) {} static OUString toUNOname(char const* p) { #if defined BRIDGE_DEBUG char const* start = p; #endif // example: N3com3sun4star4lang24IllegalArgumentExceptionE OUStringBuffer buf(64); assert('N' == *p); ++p; // skip N while ('E' != *p) { // read chars count long n = (*p++ - '0'); while ('0' <= *p && '9' >= *p) { n *= 10; n += (*p++ - '0'); } buf.appendAscii(p, n); p += n; if ('E' != *p) buf.append('.'); } #if defined BRIDGE_DEBUG OUString ret(buf.makeStringAndClear()); OString c_ret(OUStringToOString(ret, RTL_TEXTENCODING_ASCII_US)); fprintf(stderr, "> toUNOname(): %s => %s\n", start, c_ret.getStr()); return ret; #else return buf.makeStringAndClear(); #endif } class RTTI { typedef std::unordered_map t_rtti_map; Mutex m_mutex; t_rtti_map m_rttis; t_rtti_map m_generatedRttis; void* m_hApp; public: RTTI(); ~RTTI(); type_info* getRTTI(typelib_CompoundTypeDescription*); }; RTTI::RTTI() : m_hApp(dlopen(0, RTLD_LAZY)) { } RTTI::~RTTI() { dlclose(m_hApp); } type_info* RTTI::getRTTI(typelib_CompoundTypeDescription* pTypeDescr) { type_info* rtti; OUString const& unoName = *(OUString const*)&pTypeDescr->aBase.pTypeName; MutexGuard guard(m_mutex); t_rtti_map::const_iterator iRttiFind(m_rttis.find(unoName)); if (iRttiFind == m_rttis.end()) { // RTTI symbol OStringBuffer buf(64); buf.append("_ZTIN"); sal_Int32 index = 0; do { OUString token(unoName.getToken(0, '.', index)); buf.append(token.getLength()); OString c_token(OUStringToOString(token, RTL_TEXTENCODING_ASCII_US)); buf.append(c_token); } while (index >= 0); buf.append('E'); OString symName(buf.makeStringAndClear()); rtti = (type_info*)dlsym(m_hApp, symName.getStr()); if (rtti) { pair insertion( m_rttis.insert(t_rtti_map::value_type(unoName, rtti))); assert(insertion.second && "### inserting new rtti failed?!"); } else { // try to lookup the symbol in the generated rtti map t_rtti_map::const_iterator iFind(m_generatedRttis.find(unoName)); if (iFind == m_generatedRttis.end()) { // we must generate it ! // symbol and rtti-name is nearly identical, // the symbol is prefixed with _ZTI char const* rttiName = symName.getStr() + 4; #if defined BRIDGE_DEBUG fprintf(stderr, "generated rtti for %s\n", rttiName); #endif if (pTypeDescr->pBaseTypeDescription) { // ensure availability of base type_info* base_rtti = getRTTI( (typelib_CompoundTypeDescription*)pTypeDescr->pBaseTypeDescription); rtti = new __si_class_type_info(strdup(rttiName), (__class_type_info*)base_rtti); } else { // this class has no base class rtti = new __class_type_info(strdup(rttiName)); } pair insertion( m_generatedRttis.insert(t_rtti_map::value_type(unoName, rtti))); assert(insertion.second && "### inserting new generated rtti failed?!"); } else // taking already generated rtti { rtti = iFind->second; } } } else { rtti = iRttiFind->second; } return rtti; } static void deleteException(void* pExc) { #if defined BRIDGE_DEBUG fprintf(stderr, "in deleteException: pExc = %p\n", pExc); #endif __cxa_exception const* header = ((__cxa_exception const*)pExc - 1); typelib_TypeDescription* pTD = 0; OUString unoName(toUNOname(header->exceptionType->name())); ::typelib_typedescription_getByName(&pTD, unoName.pData); assert(pTD && "### unknown exception type! leaving out destruction => leaking!!!"); if (pTD) { ::uno_destructData(pExc, pTD, cpp_release); ::typelib_typedescription_release(pTD); } } //extern "C" { // void __cxa_throw(void* ex, void* info, void (*dest)(void*)) { ::abort(); } //} void raiseException(uno_Any* pUnoExc, uno_Mapping* pUno2Cpp) { #if defined BRIDGE_DEBUG OString cstr(OUStringToOString(OUString::unacquired(&pUnoExc->pType->pTypeName), RTL_TEXTENCODING_ASCII_US)); fprintf(stderr, "> uno exception occurred: %s\n", cstr.getStr()); #endif void* pCppExc; type_info* rtti; { // construct cpp exception object typelib_TypeDescription* pTypeDescr = 0; TYPELIB_DANGER_GET(&pTypeDescr, pUnoExc->pType); assert(pTypeDescr); if (!pTypeDescr) { throw RuntimeException(OUString("cannot get typedescription for type ") + OUString::unacquired(&pUnoExc->pType->pTypeName)); } pCppExc = __cxa_allocate_exception(pTypeDescr->nSize); ::uno_copyAndConvertData(pCppExc, pUnoExc->pData, pTypeDescr, pUno2Cpp); // destruct uno exception ::uno_any_destruct(pUnoExc, 0); // avoiding locked counts static RTTI rtti_data; rtti = (type_info*)rtti_data.getRTTI((typelib_CompoundTypeDescription*)pTypeDescr); TYPELIB_DANGER_RELEASE(pTypeDescr); assert(rtti && "### no rtti for throwing exception!"); if (!rtti) { throw RuntimeException(OUString("no rtti for type ") + OUString::unacquired(&pUnoExc->pType->pTypeName)); } } __cxa_throw(pCppExc, rtti, deleteException); } void fillUnoException(uno_Any* pUnoExc, uno_Mapping* pCpp2Uno) { __cxa_exception* header = __cxa_get_globals()->caughtExceptions; if (!header) { RuntimeException aRE("no exception header!"); Type const& rType = cppu::UnoType::get(); uno_type_any_constructAndConvert(pUnoExc, &aRE, rType.getTypeLibType(), pCpp2Uno); SAL_WARN("bridges", aRE.Message); return; } std::type_info* exceptionType = __cxa_current_exception_type(); typelib_TypeDescription* pExcTypeDescr = 0; OUString unoName(toUNOname(exceptionType->name())); #if defined BRIDGE_DEBUG OString cstr_unoName(OUStringToOString(unoName, RTL_TEXTENCODING_ASCII_US)); fprintf(stderr, "> c++ exception occurred: %s\n", cstr_unoName.getStr()); #endif typelib_typedescription_getByName(&pExcTypeDescr, unoName.pData); if (0 == pExcTypeDescr) { RuntimeException aRE(OUString("exception type not found: ") + unoName); Type const& rType = cppu::UnoType::get(); uno_type_any_constructAndConvert(pUnoExc, &aRE, rType.getTypeLibType(), pCpp2Uno); SAL_WARN("bridges", aRE.Message); } else { // construct uno exception any uno_any_constructAndConvert(pUnoExc, header->adjustedPtr, pExcTypeDescr, pCpp2Uno); typelib_typedescription_release(pExcTypeDescr); } } } /* vim:set shiftwidth=4 softtabstop=4 expandtab cinoptions=b1,g0,N-s cinkeys+=0=break: */