/* -*- Mode: C++; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4 -*- */ /* * 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 #include #include #include #include #include #include #include #include #include #include using namespace osl; using namespace com::sun::star; using namespace com::sun::star::uno; using namespace com::sun::star::lang; using namespace com::sun::star::loader; using namespace com::sun::star::registry; using ::rtl::OUString; namespace cppu { class OSingleFactoryHelper : public XServiceInfo , public XSingleServiceFactory , public lang::XSingleComponentFactory , public XUnloadingPreference { public: OSingleFactoryHelper( const Reference & rServiceManager, const OUString & rImplementationName_, ComponentInstantiation pCreateFunction_, ComponentFactoryFunc fptr, const Sequence< OUString > * pServiceNames_ ) : xSMgr( rServiceManager ) , pCreateFunction( pCreateFunction_ ) , m_fptr( fptr ) , aImplementationName( rImplementationName_ ) { if( pServiceNames_ ) aServiceNames = *pServiceNames_; } virtual ~OSingleFactoryHelper(); // XInterface Any SAL_CALL queryInterface( const Type & rType ) throw(::com::sun::star::uno::RuntimeException, std::exception) SAL_OVERRIDE; // XSingleServiceFactory Reference SAL_CALL createInstance() throw(::com::sun::star::uno::Exception, ::com::sun::star::uno::RuntimeException, std::exception) SAL_OVERRIDE; virtual Reference SAL_CALL createInstanceWithArguments(const Sequence& Arguments) throw(::com::sun::star::uno::Exception, ::com::sun::star::uno::RuntimeException, std::exception) SAL_OVERRIDE; // XSingleComponentFactory virtual Reference< XInterface > SAL_CALL createInstanceWithContext( Reference< XComponentContext > const & xContext ) throw (Exception, RuntimeException, std::exception) SAL_OVERRIDE; virtual Reference< XInterface > SAL_CALL createInstanceWithArgumentsAndContext( Sequence< Any > const & rArguments, Reference< XComponentContext > const & xContext ) throw (Exception, RuntimeException, std::exception) SAL_OVERRIDE; // XServiceInfo OUString SAL_CALL getImplementationName() throw(::com::sun::star::uno::RuntimeException, std::exception) SAL_OVERRIDE; sal_Bool SAL_CALL supportsService(const OUString& ServiceName) throw(::com::sun::star::uno::RuntimeException, std::exception) SAL_OVERRIDE; Sequence< OUString > SAL_CALL getSupportedServiceNames(void) throw(::com::sun::star::uno::RuntimeException, std::exception) SAL_OVERRIDE; protected: /** * Create an instance specified by the factory. The one instance logic is implemented * in the createInstance and createInstanceWithArguments methods. * @return the newly created instance. Do not return a previous (one instance) instance. */ virtual Reference createInstanceEveryTime( Reference< XComponentContext > const & xContext ) throw(::com::sun::star::uno::Exception, ::com::sun::star::uno::RuntimeException); Reference xSMgr; ComponentInstantiation pCreateFunction; ComponentFactoryFunc m_fptr; Sequence< OUString > aServiceNames; OUString aImplementationName; }; OSingleFactoryHelper::~OSingleFactoryHelper() { } Any OSingleFactoryHelper::queryInterface( const Type & rType ) throw(::com::sun::star::uno::RuntimeException, std::exception) { return ::cppu::queryInterface( rType, static_cast< XSingleComponentFactory * >( this ), static_cast< XSingleServiceFactory * >( this ), static_cast< XServiceInfo * >( this ) , static_cast< XUnloadingPreference * >( this )); } // OSingleFactoryHelper Reference OSingleFactoryHelper::createInstanceEveryTime( Reference< XComponentContext > const & xContext ) throw(::com::sun::star::uno::Exception, ::com::sun::star::uno::RuntimeException) { if (m_fptr) { return (*m_fptr)( xContext ); } else if( pCreateFunction ) { if (xContext.is()) { Reference< lang::XMultiServiceFactory > xContextMgr( xContext->getServiceManager(), UNO_QUERY ); if (xContextMgr.is()) return (*pCreateFunction)( xContextMgr ); } return (*pCreateFunction)( xSMgr ); } else { return Reference< XInterface >(); } } // XSingleServiceFactory Reference OSingleFactoryHelper::createInstance() throw(::com::sun::star::uno::Exception, ::com::sun::star::uno::RuntimeException, std::exception) { return createInstanceWithContext( Reference< XComponentContext >() ); } // XSingleServiceFactory Reference OSingleFactoryHelper::createInstanceWithArguments( const Sequence& Arguments ) throw(::com::sun::star::uno::Exception, ::com::sun::star::uno::RuntimeException, std::exception) { return createInstanceWithArgumentsAndContext( Arguments, Reference< XComponentContext >() ); } // XSingleComponentFactory Reference< XInterface > OSingleFactoryHelper::createInstanceWithContext( Reference< XComponentContext > const & xContext ) throw (Exception, RuntimeException, std::exception) { return createInstanceEveryTime( xContext ); } Reference< XInterface > OSingleFactoryHelper::createInstanceWithArgumentsAndContext( Sequence< Any > const & rArguments, Reference< XComponentContext > const & xContext ) throw (Exception, RuntimeException, std::exception) { Reference< XInterface > xRet( createInstanceWithContext( xContext ) ); Reference< lang::XInitialization > xInit( xRet, UNO_QUERY ); // always call initialize, even if there are no arguments. // #i63511# / 2006-03-27 / frank.schoenheit@sun.com if (xInit.is()) { xInit->initialize( rArguments ); } else { if ( rArguments.getLength() ) { // dispose the here created UNO object before throwing out exception // to avoid risk of memory leaks #i113722# Reference xComp( xRet, UNO_QUERY ); if (xComp.is()) xComp->dispose(); throw lang::IllegalArgumentException( OUString("cannot pass arguments to component => no XInitialization implemented!"), Reference< XInterface >(), 0 ); } } return xRet; } // XServiceInfo OUString OSingleFactoryHelper::getImplementationName() throw(::com::sun::star::uno::RuntimeException, std::exception) { return aImplementationName; } // XServiceInfo sal_Bool OSingleFactoryHelper::supportsService( const OUString& ServiceName ) throw(::com::sun::star::uno::RuntimeException, std::exception) { return cppu::supportsService(this, ServiceName); } // XServiceInfo Sequence< OUString > OSingleFactoryHelper::getSupportedServiceNames(void) throw(::com::sun::star::uno::RuntimeException, std::exception) { return aServiceNames; } struct OFactoryComponentHelper_Mutex { Mutex aMutex; }; class OFactoryComponentHelper : public OFactoryComponentHelper_Mutex , public OComponentHelper , public OSingleFactoryHelper { public: OFactoryComponentHelper( const Reference & rServiceManager, const OUString & rImplementationName_, ComponentInstantiation pCreateFunction_, ComponentFactoryFunc fptr, const Sequence< OUString > * pServiceNames_, bool bOneInstance_ = false ) : OComponentHelper( aMutex ) , OSingleFactoryHelper( rServiceManager, rImplementationName_, pCreateFunction_, fptr, pServiceNames_ ) , bOneInstance( bOneInstance_ ) { } // XInterface Any SAL_CALL queryInterface( const Type & rType ) throw(::com::sun::star::uno::RuntimeException, std::exception) SAL_OVERRIDE; void SAL_CALL acquire() throw() SAL_OVERRIDE { OComponentHelper::acquire(); } void SAL_CALL release() throw() SAL_OVERRIDE { OComponentHelper::release(); } // XSingleServiceFactory Reference SAL_CALL createInstance() throw(::com::sun::star::uno::Exception, ::com::sun::star::uno::RuntimeException, std::exception) SAL_OVERRIDE; Reference SAL_CALL createInstanceWithArguments( const Sequence& Arguments ) throw(::com::sun::star::uno::Exception, ::com::sun::star::uno::RuntimeException, std::exception) SAL_OVERRIDE; // XSingleComponentFactory virtual Reference< XInterface > SAL_CALL createInstanceWithContext( Reference< XComponentContext > const & xContext ) throw (Exception, RuntimeException, std::exception) SAL_OVERRIDE; virtual Reference< XInterface > SAL_CALL createInstanceWithArgumentsAndContext( Sequence< Any > const & rArguments, Reference< XComponentContext > const & xContext ) throw (Exception, RuntimeException, std::exception) SAL_OVERRIDE; // XTypeProvider virtual Sequence< Type > SAL_CALL getTypes() throw (::com::sun::star::uno::RuntimeException, std::exception) SAL_OVERRIDE; virtual Sequence< sal_Int8 > SAL_CALL getImplementationId() throw(::com::sun::star::uno::RuntimeException, std::exception) SAL_OVERRIDE; // XAggregation Any SAL_CALL queryAggregation( const Type & rType ) throw(::com::sun::star::uno::RuntimeException, std::exception) SAL_OVERRIDE; // XUnloadingPreference virtual sal_Bool SAL_CALL releaseOnNotification() throw(::com::sun::star::uno::RuntimeException, std::exception) SAL_OVERRIDE; // OComponentHelper void SAL_CALL dispose() throw(::com::sun::star::uno::RuntimeException, std::exception) SAL_OVERRIDE; private: Reference xTheInstance; bool bOneInstance; protected: // needed for implementing XUnloadingPreference in inheriting classes bool isOneInstance() {return bOneInstance;} bool isInstance() {return xTheInstance.is();} }; Any SAL_CALL OFactoryComponentHelper::queryInterface( const Type & rType ) throw(::com::sun::star::uno::RuntimeException, std::exception) { if( rType == ::getCppuType( (Reference*)0)) { return makeAny( Reference< XUnloadingPreference >( static_cast< XUnloadingPreference * >(this) ) ); } return OComponentHelper::queryInterface( rType ); } // XAggregation Any OFactoryComponentHelper::queryAggregation( const Type & rType ) throw(::com::sun::star::uno::RuntimeException, std::exception) { Any aRet( OComponentHelper::queryAggregation( rType ) ); return (aRet.hasValue() ? aRet : OSingleFactoryHelper::queryInterface( rType )); } // XTypeProvider Sequence< Type > OFactoryComponentHelper::getTypes() throw (::com::sun::star::uno::RuntimeException, std::exception) { Type ar[ 4 ]; ar[ 0 ] = ::getCppuType( (const Reference< XSingleServiceFactory > *)0 ); ar[ 1 ] = ::getCppuType( (const Reference< XServiceInfo > *)0 ); ar[ 2 ] = ::getCppuType( (const Reference< XUnloadingPreference > *)0 ); if (m_fptr) ar[ 3 ] = ::getCppuType( (const Reference< XSingleComponentFactory > *)0 ); return Sequence< Type >( ar, m_fptr ? 4 : 3 ); } Sequence< sal_Int8 > OFactoryComponentHelper::getImplementationId() throw (::com::sun::star::uno::RuntimeException, std::exception) { return css::uno::Sequence(); } // XSingleServiceFactory Reference OFactoryComponentHelper::createInstance() throw(::com::sun::star::uno::Exception, ::com::sun::star::uno::RuntimeException, std::exception) { if( bOneInstance ) { if( !xTheInstance.is() ) { MutexGuard aGuard( aMutex ); if( !xTheInstance.is() ) xTheInstance = OSingleFactoryHelper::createInstance(); } return xTheInstance; } return OSingleFactoryHelper::createInstance(); } Reference OFactoryComponentHelper::createInstanceWithArguments( const Sequence& Arguments ) throw(::com::sun::star::uno::Exception, ::com::sun::star::uno::RuntimeException, std::exception) { if( bOneInstance ) { if( !xTheInstance.is() ) { MutexGuard aGuard( aMutex ); // OSL_ENSURE( !xTheInstance.is(), "### arguments will be ignored!" ); if( !xTheInstance.is() ) xTheInstance = OSingleFactoryHelper::createInstanceWithArguments( Arguments ); } return xTheInstance; } return OSingleFactoryHelper::createInstanceWithArguments( Arguments ); } // XSingleComponentFactory Reference< XInterface > OFactoryComponentHelper::createInstanceWithContext( Reference< XComponentContext > const & xContext ) throw (Exception, RuntimeException, std::exception) { if( bOneInstance ) { if( !xTheInstance.is() ) { MutexGuard aGuard( aMutex ); // OSL_ENSURE( !xTheInstance.is(), "### context will be ignored!" ); if( !xTheInstance.is() ) xTheInstance = OSingleFactoryHelper::createInstanceWithContext( xContext ); } return xTheInstance; } return OSingleFactoryHelper::createInstanceWithContext( xContext ); } Reference< XInterface > OFactoryComponentHelper::createInstanceWithArgumentsAndContext( Sequence< Any > const & rArguments, Reference< XComponentContext > const & xContext ) throw (Exception, RuntimeException, std::exception) { if( bOneInstance ) { if( !xTheInstance.is() ) { MutexGuard aGuard( aMutex ); // OSL_ENSURE( !xTheInstance.is(), "### context and arguments will be ignored!" ); if( !xTheInstance.is() ) xTheInstance = OSingleFactoryHelper::createInstanceWithArgumentsAndContext( rArguments, xContext ); } return xTheInstance; } return OSingleFactoryHelper::createInstanceWithArgumentsAndContext( rArguments, xContext ); } // OComponentHelper void OFactoryComponentHelper::dispose() throw(::com::sun::star::uno::RuntimeException, std::exception) { OComponentHelper::dispose(); Reference x; { // do not delete in the guard section MutexGuard aGuard( aMutex ); x = xTheInstance; xTheInstance.clear(); } // if it is a component call dispose at the component Reference xComp( x, UNO_QUERY ); if( xComp.is() ) xComp->dispose(); } // XUnloadingPreference // This class is used for single factories, component factories and // one-instance factories. Depending on the usage this function has // to return different values. // one-instance factory: sal_False // single factory: sal_True // component factory: sal_True sal_Bool SAL_CALL OFactoryComponentHelper::releaseOnNotification() throw(::com::sun::star::uno::RuntimeException, std::exception) { if( bOneInstance) return sal_False; return sal_True; } class ORegistryFactoryHelper : public OFactoryComponentHelper, public OPropertySetHelper { public: ORegistryFactoryHelper( const Reference & rServiceManager, const OUString & rImplementationName_, const Reference & xImplementationKey_, bool bOneInstance_ = false ) : OFactoryComponentHelper( rServiceManager, rImplementationName_, 0, 0, 0, bOneInstance_ ), OPropertySetHelper( OComponentHelper::rBHelper ), xImplementationKey( xImplementationKey_ ) {} // XInterface virtual Any SAL_CALL queryInterface( Type const & type ) throw (RuntimeException, std::exception) SAL_OVERRIDE; virtual void SAL_CALL acquire() throw () SAL_OVERRIDE; virtual void SAL_CALL release() throw () SAL_OVERRIDE; // XTypeProvider virtual Sequence< Type > SAL_CALL getTypes() throw (RuntimeException, std::exception) SAL_OVERRIDE; // XPropertySet virtual Reference< beans::XPropertySetInfo > SAL_CALL getPropertySetInfo() throw (RuntimeException, std::exception) SAL_OVERRIDE; // OPropertySetHelper virtual IPropertyArrayHelper & SAL_CALL getInfoHelper() SAL_OVERRIDE; virtual sal_Bool SAL_CALL convertFastPropertyValue( Any & rConvertedValue, Any & rOldValue, sal_Int32 nHandle, Any const & rValue ) throw (lang::IllegalArgumentException) SAL_OVERRIDE; virtual void SAL_CALL setFastPropertyValue_NoBroadcast( sal_Int32 nHandle, Any const & rValue ) throw (Exception, std::exception) SAL_OVERRIDE; using OPropertySetHelper::getFastPropertyValue; virtual void SAL_CALL getFastPropertyValue( Any & rValue, sal_Int32 nHandle ) const SAL_OVERRIDE; // OSingleFactoryHelper Reference createInstanceEveryTime( Reference< XComponentContext > const & xContext ) throw(::com::sun::star::uno::Exception, ::com::sun::star::uno::RuntimeException) SAL_OVERRIDE; // XSingleServiceFactory Reference SAL_CALL createInstanceWithArguments(const Sequence& Arguments) throw(::com::sun::star::uno::Exception, ::com::sun::star::uno::RuntimeException, std::exception) SAL_OVERRIDE; // XSingleComponentFactory Reference< XInterface > SAL_CALL createInstanceWithArgumentsAndContext( Sequence< Any > const & rArguments, Reference< XComponentContext > const & xContext ) throw (Exception, RuntimeException, std::exception) SAL_OVERRIDE; // XServiceInfo Sequence< OUString > SAL_CALL getSupportedServiceNames(void) throw(::com::sun::star::uno::RuntimeException, std::exception) SAL_OVERRIDE; // XUnloadingPreference sal_Bool SAL_CALL releaseOnNotification() throw( RuntimeException, std::exception) SAL_OVERRIDE; private: Reference< XInterface > createModuleFactory() throw(::com::sun::star::uno::Exception, ::com::sun::star::uno::RuntimeException); /** The registry key of the implementation section */ Reference xImplementationKey; /** The factory created with the loader. */ Reference xModuleFactory; Reference xModuleFactoryDepr; Reference< beans::XPropertySetInfo > m_xInfo; ::std::unique_ptr< IPropertyArrayHelper > m_property_array_helper; protected: using OPropertySetHelper::getTypes; }; // XInterface Any SAL_CALL ORegistryFactoryHelper::queryInterface( Type const & type ) throw (RuntimeException, std::exception) { Any ret( OFactoryComponentHelper::queryInterface( type ) ); if (ret.hasValue()) return ret; else return OPropertySetHelper::queryInterface( type ); } void ORegistryFactoryHelper::acquire() throw () { OFactoryComponentHelper::acquire(); } void ORegistryFactoryHelper::release() throw () { OFactoryComponentHelper::release(); } // XTypeProvider Sequence< Type > ORegistryFactoryHelper::getTypes() throw (RuntimeException, std::exception) { Sequence< Type > types( OFactoryComponentHelper::getTypes() ); sal_Int32 pos = types.getLength(); types.realloc( pos + 3 ); Type * p = types.getArray(); p[ pos++ ] = ::getCppuType( reinterpret_cast< Reference< beans::XMultiPropertySet > const * >(0) ); p[ pos++ ] = ::getCppuType( reinterpret_cast< Reference< beans::XFastPropertySet > const * >(0) ); p[ pos++ ] = ::getCppuType( reinterpret_cast< Reference< beans::XPropertySet > const * >(0) ); return types; } // XPropertySet Reference< beans::XPropertySetInfo > ORegistryFactoryHelper::getPropertySetInfo() throw (RuntimeException, std::exception) { ::osl::MutexGuard guard( aMutex ); if (! m_xInfo.is()) m_xInfo = createPropertySetInfo( getInfoHelper() ); return m_xInfo; } // OPropertySetHelper IPropertyArrayHelper & ORegistryFactoryHelper::getInfoHelper() { ::osl::MutexGuard guard( aMutex ); if (m_property_array_helper.get() == 0) { beans::Property prop( "ImplementationKey" /* name */, 0 /* handle */, ::getCppuType( &xImplementationKey ), beans::PropertyAttribute::READONLY | beans::PropertyAttribute::OPTIONAL ); m_property_array_helper.reset( new ::cppu::OPropertyArrayHelper( &prop, 1 ) ); } return *m_property_array_helper.get(); } sal_Bool ORegistryFactoryHelper::convertFastPropertyValue( Any &, Any &, sal_Int32, Any const & ) throw (lang::IllegalArgumentException) { OSL_FAIL( "unexpected!" ); return false; } void ORegistryFactoryHelper::setFastPropertyValue_NoBroadcast( sal_Int32, Any const & ) throw (Exception, std::exception) { throw beans::PropertyVetoException( "unexpected: only readonly properties!", static_cast< OWeakObject * >(this) ); } void ORegistryFactoryHelper::getFastPropertyValue( Any & rValue, sal_Int32 nHandle ) const { if (nHandle == 0) { rValue <<= xImplementationKey; } else { rValue.clear(); throw beans::UnknownPropertyException( "unknown property!", static_cast< OWeakObject * >( const_cast< ORegistryFactoryHelper * >(this) ) ); } } Reference ORegistryFactoryHelper::createInstanceEveryTime( Reference< XComponentContext > const & xContext ) throw(::com::sun::star::uno::Exception, ::com::sun::star::uno::RuntimeException) { if( !xModuleFactory.is() && !xModuleFactoryDepr.is() ) { Reference< XInterface > x( createModuleFactory() ); if (x.is()) { MutexGuard aGuard( aMutex ); if( !xModuleFactory.is() && !xModuleFactoryDepr.is() ) { xModuleFactory.set( x, UNO_QUERY ); xModuleFactoryDepr.set( x, UNO_QUERY ); } } } if( xModuleFactory.is() ) { return xModuleFactory->createInstanceWithContext( xContext ); } else if( xModuleFactoryDepr.is() ) { return xModuleFactoryDepr->createInstance(); } return Reference(); } Reference SAL_CALL ORegistryFactoryHelper::createInstanceWithArguments( const Sequence& Arguments ) throw(::com::sun::star::uno::Exception, ::com::sun::star::uno::RuntimeException, std::exception) { if( !xModuleFactory.is() && !xModuleFactoryDepr.is() ) { Reference< XInterface > x( createModuleFactory() ); if (x.is()) { MutexGuard aGuard( aMutex ); if( !xModuleFactory.is() && !xModuleFactoryDepr.is() ) { xModuleFactory.set( x, UNO_QUERY ); xModuleFactoryDepr.set( x, UNO_QUERY ); } } } if( xModuleFactoryDepr.is() ) { return xModuleFactoryDepr->createInstanceWithArguments( Arguments ); } else if( xModuleFactory.is() ) { #if OSL_DEBUG_LEVEL > 1 OSL_TRACE( "### no context ORegistryFactoryHelper::createInstanceWithArgumentsAndContext()!" ); #endif return xModuleFactory->createInstanceWithArgumentsAndContext( Arguments, Reference< XComponentContext >() ); } return Reference(); } Reference< XInterface > ORegistryFactoryHelper::createInstanceWithArgumentsAndContext( Sequence< Any > const & rArguments, Reference< XComponentContext > const & xContext ) throw (Exception, RuntimeException, std::exception) { if( !xModuleFactory.is() && !xModuleFactoryDepr.is() ) { Reference< XInterface > x( createModuleFactory() ); if (x.is()) { MutexGuard aGuard( aMutex ); if( !xModuleFactory.is() && !xModuleFactoryDepr.is() ) { xModuleFactory.set( x, UNO_QUERY ); xModuleFactoryDepr.set( x, UNO_QUERY ); } } } if( xModuleFactory.is() ) { return xModuleFactory->createInstanceWithArgumentsAndContext( rArguments, xContext ); } else if( xModuleFactoryDepr.is() ) { #if OSL_DEBUG_LEVEL > 1 if (xContext.is()) { OSL_TRACE( "### ignoring context calling ORegistryFactoryHelper::createInstanceWithArgumentsAndContext()!" ); } #endif return xModuleFactoryDepr->createInstanceWithArguments( rArguments ); } return Reference(); } // OSingleFactoryHelper Reference< XInterface > ORegistryFactoryHelper::createModuleFactory() throw(::com::sun::star::uno::Exception, ::com::sun::star::uno::RuntimeException) { OUString aActivatorUrl; OUString aActivatorName; OUString aLocation; Reference xActivatorKey = xImplementationKey->openKey( OUString("/UNO/ACTIVATOR") ); if( xActivatorKey.is() && xActivatorKey->getValueType() == RegistryValueType_ASCII ) { aActivatorUrl = xActivatorKey->getAsciiValue(); OUString tmpActivator(aActivatorUrl.getStr()); sal_Int32 nIndex = 0; aActivatorName = tmpActivator.getToken(0, ':', nIndex ); Reference xLocationKey = xImplementationKey->openKey( OUString("/UNO/LOCATION") ); if( xLocationKey.is() && xLocationKey->getValueType() == RegistryValueType_ASCII ) aLocation = xLocationKey->getAsciiValue(); } else { // old style"url" // the location of the program code of the implementation Reference xLocationKey = xImplementationKey->openKey( OUString("/UNO/URL") ); // is the key of the right type ? if( xLocationKey.is() && xLocationKey->getValueType() == RegistryValueType_ASCII ) { // one implementation found -> try to activate aLocation = xLocationKey->getAsciiValue(); // search protocol delimiter sal_Int32 nPos = aLocation.indexOf( OUString("://") ); if( nPos != -1 ) { aActivatorName = aLocation.copy( 0, nPos ); if( aActivatorName == "java" ) aActivatorName = "com.sun.star.loader.Java"; else if( aActivatorName == "module" ) aActivatorName = "com.sun.star.loader.SharedLibrary"; aLocation = aLocation.copy( nPos + 3 ); } } } Reference< XInterface > xFactory; if( !aActivatorName.isEmpty() ) { Reference x = xSMgr->createInstance( aActivatorName ); Reference xLoader( x, UNO_QUERY ); Reference xMF; if (xLoader.is()) { xFactory = xLoader->activate( aImplementationName, aActivatorUrl, aLocation, xImplementationKey ); } } return xFactory; } // XServiceInfo Sequence< OUString > ORegistryFactoryHelper::getSupportedServiceNames(void) throw(::com::sun::star::uno::RuntimeException, std::exception) { MutexGuard aGuard( aMutex ); if( aServiceNames.getLength() == 0 ) { // not yet loaded try { Reference xKey = xImplementationKey->openKey( OUString("UNO/SERVICES") ); if (xKey.is()) { // length of prefix. +1 for the '/' at the end sal_Int32 nPrefixLen = xKey->getKeyName().getLength() + 1; // Full qualified names like "IMPLEMENTATIONS/TEST/UNO/SERVICES/com.sun.star..." Sequence seqKeys = xKey->getKeyNames(); OUString* pKeys = seqKeys.getArray(); for( sal_Int32 i = 0; i < seqKeys.getLength(); i++ ) pKeys[i] = pKeys[i].copy(nPrefixLen); aServiceNames = seqKeys; } } catch (InvalidRegistryException &) { } } return aServiceNames; } sal_Bool SAL_CALL ORegistryFactoryHelper::releaseOnNotification() throw(::com::sun::star::uno::RuntimeException, std::exception) { bool retVal= true; if( isOneInstance() && isInstance()) { retVal= false; } else if( ! isOneInstance()) { // try to delegate if( xModuleFactory.is()) { Reference xunloading( xModuleFactory, UNO_QUERY); if( xunloading.is()) retVal= xunloading->releaseOnNotification(); } else if( xModuleFactoryDepr.is()) { Reference xunloading( xModuleFactoryDepr, UNO_QUERY); if( xunloading.is()) retVal= xunloading->releaseOnNotification(); } } return retVal; } class OFactoryProxyHelper : public WeakImplHelper< XServiceInfo, XSingleServiceFactory, XUnloadingPreference > { Reference xFactory; public: OFactoryProxyHelper( const Reference & rFactory ) : xFactory( rFactory ) {} // XSingleServiceFactory Reference SAL_CALL createInstance() throw(::com::sun::star::uno::Exception, ::com::sun::star::uno::RuntimeException, std::exception) SAL_OVERRIDE; Reference SAL_CALL createInstanceWithArguments(const Sequence& Arguments) throw(::com::sun::star::uno::Exception, ::com::sun::star::uno::RuntimeException, std::exception) SAL_OVERRIDE; // XServiceInfo OUString SAL_CALL getImplementationName() throw(::com::sun::star::uno::RuntimeException, std::exception) SAL_OVERRIDE; sal_Bool SAL_CALL supportsService(const OUString& ServiceName) throw(::com::sun::star::uno::RuntimeException, std::exception) SAL_OVERRIDE; Sequence< OUString > SAL_CALL getSupportedServiceNames(void) throw(::com::sun::star::uno::RuntimeException, std::exception) SAL_OVERRIDE; //XUnloadingPreference sal_Bool SAL_CALL releaseOnNotification() throw(::com::sun::star::uno::RuntimeException, std::exception) SAL_OVERRIDE; }; // XSingleServiceFactory Reference OFactoryProxyHelper::createInstance() throw(::com::sun::star::uno::Exception, ::com::sun::star::uno::RuntimeException, std::exception) { return xFactory->createInstance(); } // XSingleServiceFactory Reference OFactoryProxyHelper::createInstanceWithArguments ( const Sequence& Arguments ) throw(::com::sun::star::uno::Exception, ::com::sun::star::uno::RuntimeException, std::exception) { return xFactory->createInstanceWithArguments( Arguments ); } // XServiceInfo OUString OFactoryProxyHelper::getImplementationName() throw(::com::sun::star::uno::RuntimeException, std::exception) { Reference xInfo( xFactory, UNO_QUERY ); if( xInfo.is() ) return xInfo->getImplementationName(); return OUString(); } // XServiceInfo sal_Bool OFactoryProxyHelper::supportsService(const OUString& ServiceName) throw(::com::sun::star::uno::RuntimeException, std::exception) { return cppu::supportsService(this, ServiceName); } // XServiceInfo Sequence< OUString > OFactoryProxyHelper::getSupportedServiceNames(void) throw(::com::sun::star::uno::RuntimeException, std::exception) { Reference xInfo( xFactory, UNO_QUERY ); if( xInfo.is() ) return xInfo->getSupportedServiceNames(); return Sequence< OUString >(); } sal_Bool SAL_CALL OFactoryProxyHelper::releaseOnNotification() throw(::com::sun::star::uno::RuntimeException, std::exception) { Reference pref( xFactory, UNO_QUERY); if( pref.is()) return pref->releaseOnNotification(); return sal_True; } // global function Reference SAL_CALL createSingleFactory( const Reference & rServiceManager, const OUString & rImplementationName, ComponentInstantiation pCreateFunction, const Sequence< OUString > & rServiceNames, rtl_ModuleCount * ) { return new OFactoryComponentHelper( rServiceManager, rImplementationName, pCreateFunction, 0, &rServiceNames, false ); } // global function Reference SAL_CALL createFactoryProxy( SAL_UNUSED_PARAMETER const Reference &, const Reference & rFactory ) { return new OFactoryProxyHelper( rFactory ); } // global function Reference SAL_CALL createOneInstanceFactory( const Reference & rServiceManager, const OUString & rImplementationName, ComponentInstantiation pCreateFunction, const Sequence< OUString > & rServiceNames, rtl_ModuleCount * ) { return new OFactoryComponentHelper( rServiceManager, rImplementationName, pCreateFunction, 0, &rServiceNames, true ); } // global function Reference SAL_CALL createSingleRegistryFactory( const Reference & rServiceManager, const OUString & rImplementationName, const Reference & rImplementationKey ) { return new ORegistryFactoryHelper( rServiceManager, rImplementationName, rImplementationKey, false ); } // global function Reference SAL_CALL createOneInstanceRegistryFactory( const Reference & rServiceManager, const OUString & rImplementationName, const Reference & rImplementationKey ) { return new ORegistryFactoryHelper( rServiceManager, rImplementationName, rImplementationKey, true ); } Reference< lang::XSingleComponentFactory > SAL_CALL createSingleComponentFactory( ComponentFactoryFunc fptr, OUString const & rImplementationName, Sequence< OUString > const & rServiceNames, rtl_ModuleCount *) { return new OFactoryComponentHelper( Reference< XMultiServiceFactory >(), rImplementationName, 0, fptr, &rServiceNames, false ); } Reference< lang::XSingleComponentFactory > SAL_CALL createOneInstanceComponentFactory( ComponentFactoryFunc fptr, OUString const & rImplementationName, Sequence< OUString > const & rServiceNames, rtl_ModuleCount *) { return new OFactoryComponentHelper( Reference< XMultiServiceFactory >(), rImplementationName, 0, fptr, &rServiceNames, true ); } } /* vim:set shiftwidth=4 softtabstop=4 expandtab: */