/* -*- Mode: C++; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4 -*- */ /************************************************************************* * * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. * * Copyright 2000, 2010 Oracle and/or its affiliates. * * OpenOffice.org - a multi-platform office productivity suite * * This file is part of OpenOffice.org. * * OpenOffice.org is free software: you can redistribute it and/or modify * it under the terms of the GNU Lesser General Public License version 3 * only, as published by the Free Software Foundation. * * OpenOffice.org 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 Lesser General Public License version 3 for more details * (a copy is included in the LICENSE file that accompanied this code). * * You should have received a copy of the GNU Lesser General Public License * version 3 along with OpenOffice.org. If not, see * * for a copy of the LGPLv3 License. * ************************************************************************/ #include #include #include "rtl/process.h" #include "rtl/bootstrap.hxx" #include "rtl/random.h" #include "rtl/string.hxx" #include "rtl/ustrbuf.hxx" #include "rtl/uri.hxx" #include "osl/diagnose.h" #include "osl/file.hxx" #include "osl/module.hxx" #include "osl/security.hxx" #include "osl/thread.hxx" #include "cppuhelper/shlib.hxx" #include "cppuhelper/bootstrap.hxx" #include "cppuhelper/component_context.hxx" #include "cppuhelper/access_control.hxx" #include "cppuhelper/findsofficepath.h" #include "com/sun/star/uno/XComponentContext.hpp" #include "com/sun/star/uno/XCurrentContext.hpp" #include "com/sun/star/lang/XSingleServiceFactory.hpp" #include "com/sun/star/lang/XSingleComponentFactory.hpp" #include "com/sun/star/lang/XInitialization.hpp" #include "com/sun/star/lang/XServiceInfo.hpp" #include "com/sun/star/registry/XSimpleRegistry.hpp" #include "com/sun/star/container/XSet.hpp" #include "com/sun/star/beans/PropertyValue.hpp" #include "com/sun/star/io/IOException.hpp" #include "com/sun/star/bridge/UnoUrlResolver.hpp" #include "com/sun/star/bridge/XUnoUrlResolver.hpp" #include "macro_expander.hxx" #define OUSTR(x) ::rtl::OUString( RTL_CONSTASCII_USTRINGPARAM(x) ) #define ARLEN(x) sizeof (x) / sizeof *(x) using namespace ::rtl; using namespace ::osl; using namespace ::com::sun::star; using namespace ::com::sun::star::uno; namespace css = com::sun::star; namespace cppu { OUString const & get_this_libpath() { static OUString s_path; if (s_path.isEmpty()) { OUString path; Module::getUrlFromAddress( reinterpret_cast(get_this_libpath), path ); path = path.copy( 0, path.lastIndexOf( '/' ) ); MutexGuard guard( Mutex::getGlobalMutex() ); if (s_path.isEmpty()) s_path = path; } return s_path; } Bootstrap const & get_unorc() SAL_THROW(()) { static rtlBootstrapHandle s_bstrap = 0; if (! s_bstrap) { #ifdef ANDROID // Wouldn't it be lovely to avoid this fugly hard-coding. // The problem is that the 'create_boostrap_macro_expander_factory()' // required for bootstrapping services, calls cppu::get_unorc directly // instead of re-using the BoostrapHandle from: // defaultBootstrap_InitialComponentContext // and since rtlBootstrapHandle is not ref-counted doing anything // clean here is hardish. OUString iniName( RTL_CONSTASCII_USTRINGPARAM( "file:///assets/program/unorc" ) ); #else OUString iniName( get_this_libpath() + OUSTR("/" SAL_CONFIGFILE("uno")) ); #endif rtlBootstrapHandle bstrap = rtl_bootstrap_args_open( iniName.pData ); ClearableMutexGuard guard( Mutex::getGlobalMutex() ); if (s_bstrap) { guard.clear(); rtl_bootstrap_args_close( bstrap ); } else { s_bstrap = bstrap; } } return *(Bootstrap const *)&s_bstrap; } // private forward decl Reference< lang::XMultiComponentFactory > bootstrapInitialSF( OUString const & rBootstrapPath ) SAL_THROW( (Exception) ); Reference< XComponentContext > bootstrapInitialContext( Reference< lang::XMultiComponentFactory > const & xSF, Reference< registry::XSimpleRegistry > const & types_xRegistry, Reference< registry::XSimpleRegistry > const & services_xRegistry, OUString const & rBootstrapPath, Bootstrap const & bootstrap ) SAL_THROW( (Exception) ); Reference< XComponentContext > SAL_CALL createInitialCfgComponentContext( ContextEntry_Init const * pEntries, sal_Int32 nEntries, Reference< XComponentContext > const & xDelegate ) SAL_THROW(()); Reference< registry::XSimpleRegistry > SAL_CALL createRegistryWrapper( const Reference< XComponentContext >& xContext ); namespace { template< class T > inline beans::PropertyValue createPropertyValue( OUString const & name, T const & value ) SAL_THROW(()) { return beans::PropertyValue( name, -1, makeAny( value ), beans::PropertyState_DIRECT_VALUE ); } OUString findBootstrapArgument( const Bootstrap & bootstrap, const OUString & arg_name, sal_Bool * pFallenBack ) SAL_THROW(()) { OUString result; OUString prefixed_arg_name = OUSTR("UNO_"); prefixed_arg_name += arg_name.toAsciiUpperCase(); // environment not set -> try relative to executable if(!bootstrap.getFrom(prefixed_arg_name, result)) { if(pFallenBack) *pFallenBack = sal_True; OUString fileName; bootstrap.getIniName(fileName); // cut the rc extension OUStringBuffer result_buf( 64 ); result_buf.append( fileName.copy( 0, fileName.getLength() - strlen(SAL_CONFIGFILE(""))) ); result_buf.appendAscii( RTL_CONSTASCII_STRINGPARAM("_") ); result_buf.append( arg_name.toAsciiLowerCase() ); result_buf.appendAscii( RTL_CONSTASCII_STRINGPARAM(".rdb") ); result = result_buf.makeStringAndClear(); OSL_TRACE( (OSL_LOG_PREFIX "findBootstrapArgument, setting \"%s\" relative to" " executable \"%s\""), OUStringToOString(arg_name, RTL_TEXTENCODING_UTF8).getStr(), OUStringToOString(result, RTL_TEXTENCODING_UTF8).getStr()); } else { if(pFallenBack) *pFallenBack = sal_False; OSL_TRACE( OSL_LOG_PREFIX "findBootstrapArgument, found \"%s\" in env \"%s\"", (OUStringToOString(prefixed_arg_name, RTL_TEXTENCODING_UTF8). getStr()), OUStringToOString(result, RTL_TEXTENCODING_UTF8).getStr()); } return result; } css::uno::Reference< css::registry::XSimpleRegistry > readRdbFile( rtl::OUString const & url, bool fatalErrors, css::uno::Reference< css::registry::XSimpleRegistry > const & lastRegistry, css::uno::Reference< css::lang::XSingleServiceFactory > const & simpleRegistryFactory, css::uno::Reference< css::lang::XSingleServiceFactory > const & nestedRegistryFactory) { OSL_ASSERT(simpleRegistryFactory.is() && nestedRegistryFactory.is()); try { css::uno::Reference< css::registry::XSimpleRegistry > simple( simpleRegistryFactory->createInstance(), css::uno::UNO_QUERY_THROW); simple->open(url, true, false); if (lastRegistry.is()) { css::uno::Reference< css::registry::XSimpleRegistry > nested( nestedRegistryFactory->createInstance(), css::uno::UNO_QUERY_THROW); css::uno::Sequence< css::uno::Any > args(2); args[0] <<= lastRegistry; args[1] <<= simple; css::uno::Reference< css::lang::XInitialization >( nested, css::uno::UNO_QUERY_THROW)-> initialize(args); return nested; } else { return simple; } } catch (css::registry::InvalidRegistryException & e) { (void) e; // avoid warnings OSL_TRACE( OSL_LOG_PREFIX "warning, could not open \"%s\": \"%s\"", rtl::OUStringToOString(url, RTL_TEXTENCODING_UTF8).getStr(), rtl::OUStringToOString(e.Message, RTL_TEXTENCODING_UTF8).getStr()); if (fatalErrors) { throw; } return lastRegistry; } } Reference< registry::XSimpleRegistry > readRdbDirectory( rtl::OUString const & url, bool fatalErrors, css::uno::Reference< css::registry::XSimpleRegistry > const & lastRegistry, css::uno::Reference< css::lang::XSingleServiceFactory > const & simpleRegistryFactory, css::uno::Reference< css::lang::XSingleServiceFactory > const & nestedRegistryFactory) { OSL_ASSERT(simpleRegistryFactory.is() && nestedRegistryFactory.is()); osl::Directory dir(url); switch (dir.open()) { case osl::FileBase::E_None: break; case osl::FileBase::E_NOENT: if (!fatalErrors) { return lastRegistry; } // fall through default: throw css::uno::RuntimeException( (rtl::OUString( RTL_CONSTASCII_USTRINGPARAM("cannot open directory ")) + url), css::uno::Reference< css::uno::XInterface >()); } for (css::uno::Reference< css::registry::XSimpleRegistry > last( lastRegistry);;) { osl::DirectoryItem i; switch (dir.getNextItem(i, SAL_MAX_UINT32)) { case osl::FileBase::E_None: break; case osl::FileBase::E_NOENT: return last; default: throw css::uno::RuntimeException( (rtl::OUString( RTL_CONSTASCII_USTRINGPARAM("cannot iterate directory ")) + url), css::uno::Reference< css::uno::XInterface >()); } osl::FileStatus stat( osl_FileStatus_Mask_Type | osl_FileStatus_Mask_FileName | osl_FileStatus_Mask_FileURL); if (i.getFileStatus(stat) != osl::FileBase::E_None) { throw css::uno::RuntimeException( (rtl::OUString( RTL_CONSTASCII_USTRINGPARAM("cannot stat in directory ")) + url), css::uno::Reference< css::uno::XInterface >()); } rtl::OUString aName = stat.getFileName(); // Ignore backup files - to allow people to edit their // services/ without extremely confusing behaviour if (aName.toChar() == '.' || aName.endsWithAsciiL("~", 1)) continue; if (stat.getFileType() != osl::FileStatus::Directory) { //TODO: symlinks last = readRdbFile( stat.getFileURL(), fatalErrors, last, simpleRegistryFactory, nestedRegistryFactory); } } } Reference< registry::XSimpleRegistry > nestRegistries( const OUString &baseDir, const Reference< lang::XSingleServiceFactory > & xSimRegFac, const Reference< lang::XSingleServiceFactory > & xNesRegFac, OUString csl_rdbs, const OUString & write_rdb, sal_Bool forceWrite_rdb, sal_Bool bFallenBack ) SAL_THROW((Exception)) { sal_Int32 index; Reference< registry::XSimpleRegistry > lastRegistry; if(!write_rdb.isEmpty()) // is there a write registry given? { lastRegistry.set(xSimRegFac->createInstance(), UNO_QUERY); try { lastRegistry->open(write_rdb, sal_False, forceWrite_rdb); } catch (registry::InvalidRegistryException & e) { (void) e; // avoid warnings OSL_TRACE( OSL_LOG_PREFIX "warning, could not open \"%s\": \"%s\"", OUStringToOString(write_rdb, RTL_TEXTENCODING_UTF8).getStr(), OUStringToOString(e.Message, RTL_TEXTENCODING_UTF8).getStr()); } if(!lastRegistry->isValid()) lastRegistry.clear(); } do { index = csl_rdbs.indexOf((sal_Unicode)' '); OUString rdb_name = (index == -1) ? csl_rdbs : csl_rdbs.copy(0, index); csl_rdbs = (index == -1) ? OUString() : csl_rdbs.copy(index + 1); if (rdb_name.isEmpty()) { continue; } bool fatalErrors = !bFallenBack; if (rdb_name[0] == '?') { rdb_name = rdb_name.copy(1); fatalErrors = false; } bool directory = rdb_name.getLength() >= 3 && rdb_name[0] == '<' && rdb_name[rdb_name.getLength() - 2] == '>' && rdb_name[rdb_name.getLength() -1] == '*'; if (directory) { rdb_name = rdb_name.copy(1, rdb_name.getLength() - 3); } osl::FileBase::getAbsoluteFileURL(baseDir, rdb_name, rdb_name); lastRegistry = directory ? readRdbDirectory( rdb_name, fatalErrors, lastRegistry, xSimRegFac, xNesRegFac) : readRdbFile( rdb_name, fatalErrors, lastRegistry, xSimRegFac, xNesRegFac); } while(index != -1 && csl_rdbs.getLength()); // are there more rdbs in list? return lastRegistry; } Reference< XComponentContext > SAL_CALL defaultBootstrap_InitialComponentContext( Bootstrap const & bootstrap ) SAL_THROW( (Exception) ) { OUString bootstrapPath; if (!bootstrap.getFrom( rtl::OUString(RTL_CONSTASCII_USTRINGPARAM("URE_INTERNAL_LIB_DIR")), bootstrapPath)) { bootstrapPath = get_this_libpath(); } OUString iniDir; osl_getProcessWorkingDir(&iniDir.pData); Reference smgr_XMultiComponentFactory( bootstrapInitialSF(bootstrapPath) ); Reference smgr_XMultiServiceFactory( smgr_XMultiComponentFactory, UNO_QUERY ); Reference xEmptyKey; Reference xSimRegFac( loadSharedLibComponentFactory( OUSTR("bootstrap.uno" SAL_DLLEXTENSION), bootstrapPath, OUSTR("com.sun.star.comp.stoc.SimpleRegistry"), smgr_XMultiServiceFactory, xEmptyKey), UNO_QUERY); Reference xNesRegFac( loadSharedLibComponentFactory( OUSTR("bootstrap.uno" SAL_DLLEXTENSION), bootstrapPath, OUSTR("com.sun.star.comp.stoc.NestedRegistry"), smgr_XMultiServiceFactory, xEmptyKey), UNO_QUERY); sal_Bool bFallenback_types; OUString cls_uno_types = findBootstrapArgument( bootstrap, OUSTR("TYPES"), &bFallenback_types ); Reference types_xRegistry = nestRegistries( iniDir, xSimRegFac, xNesRegFac, cls_uno_types, OUString(), sal_False, bFallenback_types ); // ==== bootstrap from services registry ==== sal_Bool bFallenback_services; OUString cls_uno_services = findBootstrapArgument( bootstrap, OUSTR("SERVICES"), &bFallenback_services ); sal_Bool fallenBackWriteRegistry; OUString write_rdb = findBootstrapArgument( bootstrap, OUSTR("WRITERDB"), &fallenBackWriteRegistry ); if (fallenBackWriteRegistry) { // no standard write rdb anymore write_rdb = OUString(); } Reference services_xRegistry = nestRegistries( iniDir, xSimRegFac, xNesRegFac, cls_uno_services, write_rdb, !fallenBackWriteRegistry, bFallenback_services ); Reference< XComponentContext > xContext( bootstrapInitialContext( smgr_XMultiComponentFactory, types_xRegistry, services_xRegistry, bootstrapPath, bootstrap ) ); // initialize sf Reference< lang::XInitialization > xInit( smgr_XMultiComponentFactory, UNO_QUERY ); OSL_ASSERT( xInit.is() ); Sequence< Any > aSFInit( 1 ); aSFInit[ 0 ] <<= services_xRegistry; xInit->initialize( aSFInit ); return xContext; } } Reference< XComponentContext > SAL_CALL defaultBootstrap_InitialComponentContext( OUString const & iniFile ) SAL_THROW( (Exception) ) { Bootstrap bootstrap( iniFile ); if (bootstrap.getHandle() == 0) throw io::IOException(OUSTR("Cannot open for reading: ") + iniFile, 0); return defaultBootstrap_InitialComponentContext( bootstrap ); } Reference< XComponentContext > SAL_CALL defaultBootstrap_InitialComponentContext() SAL_THROW( (Exception) ) { return defaultBootstrap_InitialComponentContext( get_unorc() ); } BootstrapException::BootstrapException() { } BootstrapException::BootstrapException( const ::rtl::OUString & rMessage ) :m_aMessage( rMessage ) { } BootstrapException::BootstrapException( const BootstrapException & e ) { m_aMessage = e.m_aMessage; } BootstrapException::~BootstrapException() { } BootstrapException & BootstrapException::operator=( const BootstrapException & e ) { m_aMessage = e.m_aMessage; return *this; } const ::rtl::OUString & BootstrapException::getMessage() const { return m_aMessage; } Reference< XComponentContext > SAL_CALL bootstrap() { Reference< XComponentContext > xRemoteContext; try { char const * p1 = cppuhelper_detail_findSofficePath(); if (p1 == NULL) { throw BootstrapException( OUSTR("no soffice installation found!")); } rtl::OUString p2; if (!rtl_convertStringToUString( &p2.pData, p1, strlen(p1), osl_getThreadTextEncoding(), (RTL_TEXTTOUNICODE_FLAGS_UNDEFINED_ERROR | RTL_TEXTTOUNICODE_FLAGS_MBUNDEFINED_ERROR | RTL_TEXTTOUNICODE_FLAGS_INVALID_ERROR))) { throw BootstrapException( OUSTR("bad characters in soffice installation path!")); } OUString path; if (osl::FileBase::getFileURLFromSystemPath(p2, path) != osl::FileBase::E_None) { throw BootstrapException( OUSTR("cannot convert soffice installation path to URL!")); } if (!path.isEmpty() && path[path.getLength() - 1] != '/') { path += OUSTR("/"); } OUString uri; if (!Bootstrap::get(OUSTR("URE_BOOTSTRAP"), uri)) { Bootstrap::set( OUSTR("URE_BOOTSTRAP"), Bootstrap::encode(path + OUSTR(SAL_CONFIGFILE("fundamental")))); } // create default local component context Reference< XComponentContext > xLocalContext( defaultBootstrap_InitialComponentContext() ); if ( !xLocalContext.is() ) throw BootstrapException( OUSTR( "no local component context!" ) ); // create a random pipe name rtlRandomPool hPool = rtl_random_createPool(); if ( hPool == 0 ) throw BootstrapException( OUSTR( "cannot create random pool!" ) ); sal_uInt8 bytes[ 16 ]; if ( rtl_random_getBytes( hPool, bytes, ARLEN( bytes ) ) != rtl_Random_E_None ) throw BootstrapException( OUSTR( "random pool error!" ) ); rtl_random_destroyPool( hPool ); ::rtl::OUStringBuffer buf; buf.appendAscii( RTL_CONSTASCII_STRINGPARAM( "uno" ) ); for ( sal_uInt32 i = 0; i < ARLEN( bytes ); ++i ) buf.append( static_cast< sal_Int32 >( bytes[ i ] ) ); OUString sPipeName( buf.makeStringAndClear() ); // accept string OSL_ASSERT( buf.getLength() == 0 ); buf.appendAscii( RTL_CONSTASCII_STRINGPARAM( "--accept=pipe,name=" ) ); buf.append( sPipeName ); buf.appendAscii( RTL_CONSTASCII_STRINGPARAM( ";urp;" ) ); // arguments OUString args [] = { OUSTR( "--nologo" ), OUSTR( "--nodefault" ), OUSTR( "--norestore" ), OUSTR( "--nocrashreport" ), OUSTR( "--nolockcheck" ), buf.makeStringAndClear() }; rtl_uString * ar_args [] = { args[ 0 ].pData, args[ 1 ].pData, args[ 2 ].pData, args[ 3 ].pData, args[ 4 ].pData, args[ 5 ].pData }; ::osl::Security sec; // start office process oslProcess hProcess = 0; oslProcessError rc = osl_executeProcess( (path + OUSTR("soffice")).pData, ar_args, ARLEN( ar_args ), osl_Process_DETACHED, sec.getHandle(), 0, // => current working dir 0, 0, // => no env vars &hProcess ); switch ( rc ) { case osl_Process_E_None: osl_freeProcessHandle( hProcess ); break; case osl_Process_E_NotFound: throw BootstrapException( OUSTR( "image not found!" ) ); case osl_Process_E_TimedOut: throw BootstrapException( OUSTR( "timout occurred!" ) ); case osl_Process_E_NoPermission: throw BootstrapException( OUSTR( "permission denied!" ) ); case osl_Process_E_Unknown: throw BootstrapException( OUSTR( "unknown error!" ) ); case osl_Process_E_InvalidError: default: throw BootstrapException( OUSTR( "unmapped error!" ) ); } // create a URL resolver Reference< bridge::XUnoUrlResolver > xUrlResolver( bridge::UnoUrlResolver::create( xLocalContext ) ); // connection string OSL_ASSERT( buf.getLength() == 0 ); buf.appendAscii( RTL_CONSTASCII_STRINGPARAM( "uno:pipe,name=" ) ); buf.append( sPipeName ); buf.appendAscii( RTL_CONSTASCII_STRINGPARAM( ";urp;StarOffice.ComponentContext" ) ); OUString sConnectString( buf.makeStringAndClear() ); // wait until office is started for ( ; ; ) { try { // try to connect to office xRemoteContext.set( xUrlResolver->resolve( sConnectString ), UNO_QUERY_THROW ); break; } catch ( connection::NoConnectException & ) { // wait 500 ms, then try to connect again TimeValue tv = { 0 /* secs */, 500000000 /* nanosecs */ }; ::osl::Thread::wait( tv ); } } } catch ( Exception & e ) { throw BootstrapException( OUSTR( "unexpected UNO exception caught: " ) + e.Message ); } return xRemoteContext; } OUString bootstrap_expandUri(OUString const & uri) { static char const PREFIX[] = "vnd.sun.star.expand:"; return uri.matchAsciiL(RTL_CONSTASCII_STRINGPARAM(PREFIX)) ? cppuhelper::detail::expandMacros( rtl::Uri::decode( uri.copy(RTL_CONSTASCII_LENGTH(PREFIX)), rtl_UriDecodeWithCharset, RTL_TEXTENCODING_UTF8)) : uri; } } // namespace cppu /* vim:set shiftwidth=4 softtabstop=4 expandtab: */