1*b1cdbd2cSJim Jagielski /**************************************************************
2*b1cdbd2cSJim Jagielski  *
3*b1cdbd2cSJim Jagielski  * Licensed to the Apache Software Foundation (ASF) under one
4*b1cdbd2cSJim Jagielski  * or more contributor license agreements.  See the NOTICE file
5*b1cdbd2cSJim Jagielski  * distributed with this work for additional information
6*b1cdbd2cSJim Jagielski  * regarding copyright ownership.  The ASF licenses this file
7*b1cdbd2cSJim Jagielski  * to you under the Apache License, Version 2.0 (the
8*b1cdbd2cSJim Jagielski  * "License"); you may not use this file except in compliance
9*b1cdbd2cSJim Jagielski  * with the License.  You may obtain a copy of the License at
10*b1cdbd2cSJim Jagielski  *
11*b1cdbd2cSJim Jagielski  *   http://www.apache.org/licenses/LICENSE-2.0
12*b1cdbd2cSJim Jagielski  *
13*b1cdbd2cSJim Jagielski  * Unless required by applicable law or agreed to in writing,
14*b1cdbd2cSJim Jagielski  * software distributed under the License is distributed on an
15*b1cdbd2cSJim Jagielski  * "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
16*b1cdbd2cSJim Jagielski  * KIND, either express or implied.  See the License for the
17*b1cdbd2cSJim Jagielski  * specific language governing permissions and limitations
18*b1cdbd2cSJim Jagielski  * under the License.
19*b1cdbd2cSJim Jagielski  *
20*b1cdbd2cSJim Jagielski  *************************************************************/
21*b1cdbd2cSJim Jagielski 
22*b1cdbd2cSJim Jagielski 
23*b1cdbd2cSJim Jagielski 
24*b1cdbd2cSJim Jagielski // MARKER(update_precomp.py): autogen include statement, do not remove
25*b1cdbd2cSJim Jagielski #include "precompiled_cppuhelper.hxx"
26*b1cdbd2cSJim Jagielski 
27*b1cdbd2cSJim Jagielski #include <string.h>
28*b1cdbd2cSJim Jagielski #include <vector>
29*b1cdbd2cSJim Jagielski 
30*b1cdbd2cSJim Jagielski #include "rtl/process.h"
31*b1cdbd2cSJim Jagielski #include "rtl/bootstrap.hxx"
32*b1cdbd2cSJim Jagielski #include "rtl/random.h"
33*b1cdbd2cSJim Jagielski #include "rtl/string.hxx"
34*b1cdbd2cSJim Jagielski #include "rtl/ustrbuf.hxx"
35*b1cdbd2cSJim Jagielski #include "rtl/uri.hxx"
36*b1cdbd2cSJim Jagielski #if OSL_DEBUG_LEVEL > 0
37*b1cdbd2cSJim Jagielski #include "rtl/strbuf.hxx"
38*b1cdbd2cSJim Jagielski #endif
39*b1cdbd2cSJim Jagielski #include "osl/diagnose.h"
40*b1cdbd2cSJim Jagielski #include "osl/file.hxx"
41*b1cdbd2cSJim Jagielski #include "osl/module.hxx"
42*b1cdbd2cSJim Jagielski #include "osl/security.hxx"
43*b1cdbd2cSJim Jagielski #include "osl/thread.hxx"
44*b1cdbd2cSJim Jagielski 
45*b1cdbd2cSJim Jagielski #include "cppuhelper/shlib.hxx"
46*b1cdbd2cSJim Jagielski #include "cppuhelper/bootstrap.hxx"
47*b1cdbd2cSJim Jagielski #include "cppuhelper/component_context.hxx"
48*b1cdbd2cSJim Jagielski #include "cppuhelper/access_control.hxx"
49*b1cdbd2cSJim Jagielski #include "cppuhelper/findsofficepath.h"
50*b1cdbd2cSJim Jagielski 
51*b1cdbd2cSJim Jagielski #include <cppuhelper/com/sun/star/container/XElementAccess.hpp>
52*b1cdbd2cSJim Jagielski 
53*b1cdbd2cSJim Jagielski #include "com/sun/star/uno/XComponentContext.hpp"
54*b1cdbd2cSJim Jagielski #include "com/sun/star/uno/XCurrentContext.hpp"
55*b1cdbd2cSJim Jagielski 
56*b1cdbd2cSJim Jagielski #include "com/sun/star/lang/XSingleServiceFactory.hpp"
57*b1cdbd2cSJim Jagielski #include "com/sun/star/lang/XSingleComponentFactory.hpp"
58*b1cdbd2cSJim Jagielski #include "com/sun/star/lang/XInitialization.hpp"
59*b1cdbd2cSJim Jagielski #include "com/sun/star/lang/XServiceInfo.hpp"
60*b1cdbd2cSJim Jagielski #include "com/sun/star/registry/XSimpleRegistry.hpp"
61*b1cdbd2cSJim Jagielski #include "com/sun/star/container/XSet.hpp"
62*b1cdbd2cSJim Jagielski #include "com/sun/star/beans/PropertyValue.hpp"
63*b1cdbd2cSJim Jagielski #include "com/sun/star/io/IOException.hpp"
64*b1cdbd2cSJim Jagielski #include "com/sun/star/bridge/UnoUrlResolver.hpp"
65*b1cdbd2cSJim Jagielski #include "com/sun/star/bridge/XUnoUrlResolver.hpp"
66*b1cdbd2cSJim Jagielski 
67*b1cdbd2cSJim Jagielski #include "macro_expander.hxx"
68*b1cdbd2cSJim Jagielski 
69*b1cdbd2cSJim Jagielski #define OUSTR(x) ::rtl::OUString( RTL_CONSTASCII_USTRINGPARAM(x) )
70*b1cdbd2cSJim Jagielski #define ARLEN(x) sizeof (x) / sizeof *(x)
71*b1cdbd2cSJim Jagielski 
72*b1cdbd2cSJim Jagielski void primeWeakMap( void); // as defined in primeweak.cxx
73*b1cdbd2cSJim Jagielski 
74*b1cdbd2cSJim Jagielski using namespace ::rtl;
75*b1cdbd2cSJim Jagielski using namespace ::osl;
76*b1cdbd2cSJim Jagielski using namespace ::com::sun::star;
77*b1cdbd2cSJim Jagielski using namespace ::com::sun::star::uno;
78*b1cdbd2cSJim Jagielski 
79*b1cdbd2cSJim Jagielski namespace cppu
80*b1cdbd2cSJim Jagielski {
81*b1cdbd2cSJim Jagielski 
get_this_libpath()82*b1cdbd2cSJim Jagielski OUString const & get_this_libpath()
83*b1cdbd2cSJim Jagielski {
84*b1cdbd2cSJim Jagielski     static OUString s_path;
85*b1cdbd2cSJim Jagielski     if (0 == s_path.getLength())
86*b1cdbd2cSJim Jagielski     {
87*b1cdbd2cSJim Jagielski         OUString path;
88*b1cdbd2cSJim Jagielski         Module::getUrlFromAddress( reinterpret_cast<oslGenericFunction>(get_this_libpath), path );
89*b1cdbd2cSJim Jagielski         path = path.copy( 0, path.lastIndexOf( '/' ) );
90*b1cdbd2cSJim Jagielski         MutexGuard guard( Mutex::getGlobalMutex() );
91*b1cdbd2cSJim Jagielski         if (0 == s_path.getLength())
92*b1cdbd2cSJim Jagielski             s_path = path;
93*b1cdbd2cSJim Jagielski     }
94*b1cdbd2cSJim Jagielski     return s_path;
95*b1cdbd2cSJim Jagielski }
96*b1cdbd2cSJim Jagielski 
get_unorc()97*b1cdbd2cSJim Jagielski Bootstrap const & get_unorc() SAL_THROW( () )
98*b1cdbd2cSJim Jagielski {
99*b1cdbd2cSJim Jagielski     static rtlBootstrapHandle s_bstrap = 0;
100*b1cdbd2cSJim Jagielski     if (! s_bstrap)
101*b1cdbd2cSJim Jagielski     {
102*b1cdbd2cSJim Jagielski         OUString iniName(
103*b1cdbd2cSJim Jagielski             get_this_libpath() + OUSTR("/" SAL_CONFIGFILE("uno")) );
104*b1cdbd2cSJim Jagielski         rtlBootstrapHandle bstrap = rtl_bootstrap_args_open( iniName.pData );
105*b1cdbd2cSJim Jagielski 
106*b1cdbd2cSJim Jagielski         ClearableMutexGuard guard( Mutex::getGlobalMutex() );
107*b1cdbd2cSJim Jagielski         if (s_bstrap)
108*b1cdbd2cSJim Jagielski         {
109*b1cdbd2cSJim Jagielski             guard.clear();
110*b1cdbd2cSJim Jagielski             rtl_bootstrap_args_close( bstrap );
111*b1cdbd2cSJim Jagielski         }
112*b1cdbd2cSJim Jagielski         else
113*b1cdbd2cSJim Jagielski         {
114*b1cdbd2cSJim Jagielski             s_bstrap = bstrap;
115*b1cdbd2cSJim Jagielski         }
116*b1cdbd2cSJim Jagielski     }
117*b1cdbd2cSJim Jagielski     return *(Bootstrap const *)&s_bstrap;
118*b1cdbd2cSJim Jagielski }
119*b1cdbd2cSJim Jagielski 
120*b1cdbd2cSJim Jagielski 
addFactories(char const * const * ppNames,OUString const & bootstrapPath,Reference<lang::XMultiComponentFactory> const & xMgr,Reference<registry::XRegistryKey> const & xKey)121*b1cdbd2cSJim Jagielski void addFactories(
122*b1cdbd2cSJim Jagielski     char const * const * ppNames /* lib, implname, ..., 0 */,
123*b1cdbd2cSJim Jagielski     OUString const & bootstrapPath,
124*b1cdbd2cSJim Jagielski     Reference< lang::XMultiComponentFactory > const & xMgr,
125*b1cdbd2cSJim Jagielski     Reference< registry::XRegistryKey > const & xKey )
126*b1cdbd2cSJim Jagielski     SAL_THROW( (Exception) )
127*b1cdbd2cSJim Jagielski {
128*b1cdbd2cSJim Jagielski     Reference< container::XSet > xSet( xMgr, UNO_QUERY );
129*b1cdbd2cSJim Jagielski     OSL_ASSERT( xSet.is() );
130*b1cdbd2cSJim Jagielski     Reference< lang::XMultiServiceFactory > xSF( xMgr, UNO_QUERY );
131*b1cdbd2cSJim Jagielski 
132*b1cdbd2cSJim Jagielski     while (*ppNames)
133*b1cdbd2cSJim Jagielski     {
134*b1cdbd2cSJim Jagielski         OUString lib( OUString::createFromAscii( *ppNames++ ) );
135*b1cdbd2cSJim Jagielski         OUString implName( OUString::createFromAscii( *ppNames++ ) );
136*b1cdbd2cSJim Jagielski 
137*b1cdbd2cSJim Jagielski         Any aFac( makeAny( loadSharedLibComponentFactory(
138*b1cdbd2cSJim Jagielski                                lib, bootstrapPath, implName, xSF, xKey ) ) );
139*b1cdbd2cSJim Jagielski         xSet->insert( aFac );
140*b1cdbd2cSJim Jagielski #if OSL_DEBUG_LEVEL > 1
141*b1cdbd2cSJim Jagielski         if (xSet->has( aFac ))
142*b1cdbd2cSJim Jagielski         {
143*b1cdbd2cSJim Jagielski             Reference< lang::XServiceInfo > xInfo;
144*b1cdbd2cSJim Jagielski             if (aFac >>= xInfo)
145*b1cdbd2cSJim Jagielski             {
146*b1cdbd2cSJim Jagielski                 ::fprintf(
147*b1cdbd2cSJim Jagielski                     stderr, "> implementation %s supports: ", ppNames[ -1 ] );
148*b1cdbd2cSJim Jagielski                 Sequence< OUString > supported(
149*b1cdbd2cSJim Jagielski                     xInfo->getSupportedServiceNames() );
150*b1cdbd2cSJim Jagielski                 for ( sal_Int32 nPos = supported.getLength(); nPos--; )
151*b1cdbd2cSJim Jagielski                 {
152*b1cdbd2cSJim Jagielski                     OString str( OUStringToOString(
153*b1cdbd2cSJim Jagielski                         supported[ nPos ], RTL_TEXTENCODING_ASCII_US ) );
154*b1cdbd2cSJim Jagielski                     ::fprintf( stderr, nPos ? "%s, " : "%s\n", str.getStr() );
155*b1cdbd2cSJim Jagielski                 }
156*b1cdbd2cSJim Jagielski             }
157*b1cdbd2cSJim Jagielski             else
158*b1cdbd2cSJim Jagielski             {
159*b1cdbd2cSJim Jagielski                 ::fprintf(
160*b1cdbd2cSJim Jagielski                     stderr,
161*b1cdbd2cSJim Jagielski                     "> implementation %s provides NO lang::XServiceInfo!!!\n",
162*b1cdbd2cSJim Jagielski                     ppNames[ -1 ] );
163*b1cdbd2cSJim Jagielski             }
164*b1cdbd2cSJim Jagielski         }
165*b1cdbd2cSJim Jagielski #endif
166*b1cdbd2cSJim Jagielski #if OSL_DEBUG_LEVEL > 0
167*b1cdbd2cSJim Jagielski         if (! xSet->has( aFac ))
168*b1cdbd2cSJim Jagielski         {
169*b1cdbd2cSJim Jagielski             OStringBuffer buf( 64 );
170*b1cdbd2cSJim Jagielski             buf.append( "### failed inserting shared lib \"" );
171*b1cdbd2cSJim Jagielski             buf.append( ppNames[ -2 ] );
172*b1cdbd2cSJim Jagielski             buf.append( "\"!!!" );
173*b1cdbd2cSJim Jagielski             OString str( buf.makeStringAndClear() );
174*b1cdbd2cSJim Jagielski             OSL_ENSURE( 0, str.getStr() );
175*b1cdbd2cSJim Jagielski         }
176*b1cdbd2cSJim Jagielski #endif
177*b1cdbd2cSJim Jagielski     }
178*b1cdbd2cSJim Jagielski }
179*b1cdbd2cSJim Jagielski 
180*b1cdbd2cSJim Jagielski // private forward decl
181*b1cdbd2cSJim Jagielski Reference< lang::XMultiComponentFactory > bootstrapInitialSF(
182*b1cdbd2cSJim Jagielski     OUString const & rBootstrapPath )
183*b1cdbd2cSJim Jagielski     SAL_THROW( (Exception) );
184*b1cdbd2cSJim Jagielski 
185*b1cdbd2cSJim Jagielski Reference< XComponentContext > bootstrapInitialContext(
186*b1cdbd2cSJim Jagielski     Reference< lang::XMultiComponentFactory > const & xSF,
187*b1cdbd2cSJim Jagielski     Reference< registry::XSimpleRegistry > const & types_xRegistry,
188*b1cdbd2cSJim Jagielski     Reference< registry::XSimpleRegistry > const & services_xRegistry,
189*b1cdbd2cSJim Jagielski     OUString const & rBootstrapPath, Bootstrap const & bootstrap )
190*b1cdbd2cSJim Jagielski     SAL_THROW( (Exception) );
191*b1cdbd2cSJim Jagielski 
192*b1cdbd2cSJim Jagielski Reference< XComponentContext > SAL_CALL createInitialCfgComponentContext(
193*b1cdbd2cSJim Jagielski     ContextEntry_Init const * pEntries, sal_Int32 nEntries,
194*b1cdbd2cSJim Jagielski     Reference< XComponentContext > const & xDelegate )
195*b1cdbd2cSJim Jagielski     SAL_THROW( () );
196*b1cdbd2cSJim Jagielski 
197*b1cdbd2cSJim Jagielski Reference< registry::XSimpleRegistry > SAL_CALL createRegistryWrapper(
198*b1cdbd2cSJim Jagielski     const Reference< XComponentContext >& xContext );
199*b1cdbd2cSJim Jagielski 
200*b1cdbd2cSJim Jagielski namespace {
201*b1cdbd2cSJim Jagielski 
202*b1cdbd2cSJim Jagielski template< class T >
createPropertyValue(OUString const & name,T const & value)203*b1cdbd2cSJim Jagielski inline beans::PropertyValue createPropertyValue(
204*b1cdbd2cSJim Jagielski     OUString const & name, T const & value )
205*b1cdbd2cSJim Jagielski     SAL_THROW( () )
206*b1cdbd2cSJim Jagielski {
207*b1cdbd2cSJim Jagielski     return beans::PropertyValue(
208*b1cdbd2cSJim Jagielski         name, -1, makeAny( value ), beans::PropertyState_DIRECT_VALUE );
209*b1cdbd2cSJim Jagielski }
210*b1cdbd2cSJim Jagielski 
findBoostrapArgument(const Bootstrap & bootstrap,const OUString & arg_name,sal_Bool * pFallenBack)211*b1cdbd2cSJim Jagielski OUString findBoostrapArgument(
212*b1cdbd2cSJim Jagielski     const Bootstrap & bootstrap,
213*b1cdbd2cSJim Jagielski     const OUString & arg_name,
214*b1cdbd2cSJim Jagielski     sal_Bool * pFallenBack )
215*b1cdbd2cSJim Jagielski     SAL_THROW(())
216*b1cdbd2cSJim Jagielski {
217*b1cdbd2cSJim Jagielski     OUString result;
218*b1cdbd2cSJim Jagielski 
219*b1cdbd2cSJim Jagielski     OUString prefixed_arg_name = OUSTR("UNO_");
220*b1cdbd2cSJim Jagielski     prefixed_arg_name += arg_name.toAsciiUpperCase();
221*b1cdbd2cSJim Jagielski 
222*b1cdbd2cSJim Jagielski     // environment not set -> try relative to executable
223*b1cdbd2cSJim Jagielski     if(!bootstrap.getFrom(prefixed_arg_name, result))
224*b1cdbd2cSJim Jagielski     {
225*b1cdbd2cSJim Jagielski         if(pFallenBack)
226*b1cdbd2cSJim Jagielski             *pFallenBack = sal_True;
227*b1cdbd2cSJim Jagielski 
228*b1cdbd2cSJim Jagielski         OUString fileName;
229*b1cdbd2cSJim Jagielski         bootstrap.getIniName(fileName);
230*b1cdbd2cSJim Jagielski 
231*b1cdbd2cSJim Jagielski         // cut the rc extension
232*b1cdbd2cSJim Jagielski         OUStringBuffer result_buf( 64 );
233*b1cdbd2cSJim Jagielski         result_buf.append(
234*b1cdbd2cSJim Jagielski             fileName.copy(
235*b1cdbd2cSJim Jagielski                 0, fileName.getLength() - strlen(SAL_CONFIGFILE(""))) );
236*b1cdbd2cSJim Jagielski         result_buf.appendAscii( RTL_CONSTASCII_STRINGPARAM("_") );
237*b1cdbd2cSJim Jagielski         result_buf.append( arg_name.toAsciiLowerCase() );
238*b1cdbd2cSJim Jagielski         result_buf.appendAscii( RTL_CONSTASCII_STRINGPARAM(".rdb") );
239*b1cdbd2cSJim Jagielski         result = result_buf.makeStringAndClear();
240*b1cdbd2cSJim Jagielski 
241*b1cdbd2cSJim Jagielski #if OSL_DEBUG_LEVEL > 1
242*b1cdbd2cSJim Jagielski         OString result_dbg =
243*b1cdbd2cSJim Jagielski             OUStringToOString(result, RTL_TEXTENCODING_ASCII_US);
244*b1cdbd2cSJim Jagielski         OString arg_name_dbg =
245*b1cdbd2cSJim Jagielski             OUStringToOString(arg_name, RTL_TEXTENCODING_ASCII_US);
246*b1cdbd2cSJim Jagielski         OSL_TRACE(
247*b1cdbd2cSJim Jagielski             "cppuhelper::findBoostrapArgument - "
248*b1cdbd2cSJim Jagielski             "setting %s relative to executable: %s\n",
249*b1cdbd2cSJim Jagielski             arg_name_dbg.getStr(),
250*b1cdbd2cSJim Jagielski             result_dbg.getStr() );
251*b1cdbd2cSJim Jagielski #endif
252*b1cdbd2cSJim Jagielski     }
253*b1cdbd2cSJim Jagielski     else
254*b1cdbd2cSJim Jagielski     {
255*b1cdbd2cSJim Jagielski         if(pFallenBack)
256*b1cdbd2cSJim Jagielski             *pFallenBack = sal_False;
257*b1cdbd2cSJim Jagielski 
258*b1cdbd2cSJim Jagielski #if OSL_DEBUG_LEVEL > 1
259*b1cdbd2cSJim Jagielski         OString prefixed_arg_name_dbg = OUStringToOString(
260*b1cdbd2cSJim Jagielski             prefixed_arg_name, RTL_TEXTENCODING_ASCII_US );
261*b1cdbd2cSJim Jagielski         OString result_dbg = OUStringToOString(
262*b1cdbd2cSJim Jagielski             result, RTL_TEXTENCODING_ASCII_US );
263*b1cdbd2cSJim Jagielski         OSL_TRACE(
264*b1cdbd2cSJim Jagielski             "cppuhelper::findBoostrapArgument - found %s in env: %s",
265*b1cdbd2cSJim Jagielski             prefixed_arg_name_dbg.getStr(),
266*b1cdbd2cSJim Jagielski             result_dbg.getStr() );
267*b1cdbd2cSJim Jagielski #endif
268*b1cdbd2cSJim Jagielski     }
269*b1cdbd2cSJim Jagielski 
270*b1cdbd2cSJim Jagielski     return result;
271*b1cdbd2cSJim Jagielski }
272*b1cdbd2cSJim Jagielski 
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)273*b1cdbd2cSJim Jagielski Reference< registry::XSimpleRegistry > nestRegistries(
274*b1cdbd2cSJim Jagielski     const OUString baseDir,
275*b1cdbd2cSJim Jagielski     const Reference< lang::XSingleServiceFactory > & xSimRegFac,
276*b1cdbd2cSJim Jagielski     const Reference< lang::XSingleServiceFactory > & xNesRegFac,
277*b1cdbd2cSJim Jagielski     OUString csl_rdbs,
278*b1cdbd2cSJim Jagielski     const OUString & write_rdb,
279*b1cdbd2cSJim Jagielski     sal_Bool forceWrite_rdb,
280*b1cdbd2cSJim Jagielski     sal_Bool bFallenBack )
281*b1cdbd2cSJim Jagielski     SAL_THROW((Exception))
282*b1cdbd2cSJim Jagielski {
283*b1cdbd2cSJim Jagielski     sal_Int32 index;
284*b1cdbd2cSJim Jagielski     Reference< registry::XSimpleRegistry > lastRegistry;
285*b1cdbd2cSJim Jagielski 
286*b1cdbd2cSJim Jagielski     if(write_rdb.getLength()) // is there a write registry given?
287*b1cdbd2cSJim Jagielski     {
288*b1cdbd2cSJim Jagielski         lastRegistry.set(xSimRegFac->createInstance(), UNO_QUERY);
289*b1cdbd2cSJim Jagielski 
290*b1cdbd2cSJim Jagielski         try
291*b1cdbd2cSJim Jagielski         {
292*b1cdbd2cSJim Jagielski             lastRegistry->open(write_rdb, sal_False, forceWrite_rdb);
293*b1cdbd2cSJim Jagielski         }
294*b1cdbd2cSJim Jagielski         catch (registry::InvalidRegistryException & invalidRegistryException)
295*b1cdbd2cSJim Jagielski         {
296*b1cdbd2cSJim Jagielski             (void) invalidRegistryException;
297*b1cdbd2cSJim Jagielski #if OSL_DEBUG_LEVEL > 1
298*b1cdbd2cSJim Jagielski             OString rdb_name_tmp = OUStringToOString(
299*b1cdbd2cSJim Jagielski                 write_rdb, RTL_TEXTENCODING_ASCII_US);
300*b1cdbd2cSJim Jagielski             OString message_dbg = OUStringToOString(
301*b1cdbd2cSJim Jagielski                 invalidRegistryException.Message, RTL_TEXTENCODING_ASCII_US);
302*b1cdbd2cSJim Jagielski             OSL_TRACE(
303*b1cdbd2cSJim Jagielski                 "warning: couldn't open %s cause of %s",
304*b1cdbd2cSJim Jagielski                 rdb_name_tmp.getStr(), message_dbg.getStr() );
305*b1cdbd2cSJim Jagielski #endif
306*b1cdbd2cSJim Jagielski         }
307*b1cdbd2cSJim Jagielski 
308*b1cdbd2cSJim Jagielski         if(!lastRegistry->isValid())
309*b1cdbd2cSJim Jagielski             lastRegistry.clear();
310*b1cdbd2cSJim Jagielski     }
311*b1cdbd2cSJim Jagielski 
312*b1cdbd2cSJim Jagielski     do
313*b1cdbd2cSJim Jagielski     {
314*b1cdbd2cSJim Jagielski         index = csl_rdbs.indexOf((sal_Unicode)' ');
315*b1cdbd2cSJim Jagielski         OUString rdb_name = (index == -1) ? csl_rdbs : csl_rdbs.copy(0, index);
316*b1cdbd2cSJim Jagielski         csl_rdbs = (index == -1) ? OUString() : csl_rdbs.copy(index + 1);
317*b1cdbd2cSJim Jagielski 
318*b1cdbd2cSJim Jagielski         if (! rdb_name.getLength())
319*b1cdbd2cSJim Jagielski             continue;
320*b1cdbd2cSJim Jagielski 
321*b1cdbd2cSJim Jagielski         bool optional = ('?' == rdb_name[ 0 ]);
322*b1cdbd2cSJim Jagielski         if (optional)
323*b1cdbd2cSJim Jagielski             rdb_name = rdb_name.copy( 1 );
324*b1cdbd2cSJim Jagielski 
325*b1cdbd2cSJim Jagielski         try
326*b1cdbd2cSJim Jagielski         {
327*b1cdbd2cSJim Jagielski             Reference<registry::XSimpleRegistry> simpleRegistry(
328*b1cdbd2cSJim Jagielski                 xSimRegFac->createInstance(), UNO_QUERY_THROW );
329*b1cdbd2cSJim Jagielski 
330*b1cdbd2cSJim Jagielski             osl::FileBase::getAbsoluteFileURL(baseDir, rdb_name, rdb_name);
331*b1cdbd2cSJim Jagielski             simpleRegistry->open(rdb_name, sal_True, sal_False);
332*b1cdbd2cSJim Jagielski 
333*b1cdbd2cSJim Jagielski             if(lastRegistry.is())
334*b1cdbd2cSJim Jagielski             {
335*b1cdbd2cSJim Jagielski                 Reference< registry::XSimpleRegistry > nestedRegistry(
336*b1cdbd2cSJim Jagielski                     xNesRegFac->createInstance(), UNO_QUERY );
337*b1cdbd2cSJim Jagielski                 Reference< lang::XInitialization > nestedRegistry_xInit(
338*b1cdbd2cSJim Jagielski                     nestedRegistry, UNO_QUERY );
339*b1cdbd2cSJim Jagielski 
340*b1cdbd2cSJim Jagielski                 Sequence<Any> aArgs(2);
341*b1cdbd2cSJim Jagielski                 aArgs[0] <<= lastRegistry;
342*b1cdbd2cSJim Jagielski                 aArgs[1] <<= simpleRegistry;
343*b1cdbd2cSJim Jagielski 
344*b1cdbd2cSJim Jagielski                 nestedRegistry_xInit->initialize(aArgs);
345*b1cdbd2cSJim Jagielski 
346*b1cdbd2cSJim Jagielski                 lastRegistry = nestedRegistry;
347*b1cdbd2cSJim Jagielski             }
348*b1cdbd2cSJim Jagielski             else
349*b1cdbd2cSJim Jagielski                 lastRegistry = simpleRegistry;
350*b1cdbd2cSJim Jagielski         }
351*b1cdbd2cSJim Jagielski         catch(registry::InvalidRegistryException & invalidRegistryException)
352*b1cdbd2cSJim Jagielski         {
353*b1cdbd2cSJim Jagielski             if (! optional)
354*b1cdbd2cSJim Jagielski             {
355*b1cdbd2cSJim Jagielski                 // if a registry was explicitly given, the exception shall fly
356*b1cdbd2cSJim Jagielski                 if( ! bFallenBack )
357*b1cdbd2cSJim Jagielski                     throw;
358*b1cdbd2cSJim Jagielski             }
359*b1cdbd2cSJim Jagielski 
360*b1cdbd2cSJim Jagielski             (void) invalidRegistryException;
361*b1cdbd2cSJim Jagielski #if OSL_DEBUG_LEVEL > 1
362*b1cdbd2cSJim Jagielski             OString rdb_name_tmp = OUStringToOString(
363*b1cdbd2cSJim Jagielski                 rdb_name, RTL_TEXTENCODING_ASCII_US );
364*b1cdbd2cSJim Jagielski             OString message_dbg = OUStringToOString(
365*b1cdbd2cSJim Jagielski                 invalidRegistryException.Message, RTL_TEXTENCODING_ASCII_US );
366*b1cdbd2cSJim Jagielski             OSL_TRACE(
367*b1cdbd2cSJim Jagielski                 "warning: couldn't open %s cause of %s",
368*b1cdbd2cSJim Jagielski                 rdb_name_tmp.getStr(), message_dbg.getStr() );
369*b1cdbd2cSJim Jagielski #endif
370*b1cdbd2cSJim Jagielski         }
371*b1cdbd2cSJim Jagielski     }
372*b1cdbd2cSJim Jagielski     while(index != -1 && csl_rdbs.getLength()); // are there more rdbs in list?
373*b1cdbd2cSJim Jagielski 
374*b1cdbd2cSJim Jagielski     return lastRegistry;
375*b1cdbd2cSJim Jagielski }
376*b1cdbd2cSJim Jagielski 
377*b1cdbd2cSJim Jagielski Reference< XComponentContext >
defaultBootstrap_InitialComponentContext(Bootstrap const & bootstrap)378*b1cdbd2cSJim Jagielski SAL_CALL defaultBootstrap_InitialComponentContext(
379*b1cdbd2cSJim Jagielski     Bootstrap const & bootstrap )
380*b1cdbd2cSJim Jagielski     SAL_THROW( (Exception) )
381*b1cdbd2cSJim Jagielski {
382*b1cdbd2cSJim Jagielski     primeWeakMap();
383*b1cdbd2cSJim Jagielski 
384*b1cdbd2cSJim Jagielski     OUString bootstrapPath;
385*b1cdbd2cSJim Jagielski     if (!bootstrap.getFrom(
386*b1cdbd2cSJim Jagielski             rtl::OUString(RTL_CONSTASCII_USTRINGPARAM("URE_INTERNAL_LIB_DIR")),
387*b1cdbd2cSJim Jagielski             bootstrapPath))
388*b1cdbd2cSJim Jagielski     {
389*b1cdbd2cSJim Jagielski         bootstrapPath = get_this_libpath();
390*b1cdbd2cSJim Jagielski     }
391*b1cdbd2cSJim Jagielski 
392*b1cdbd2cSJim Jagielski     OUString iniDir;
393*b1cdbd2cSJim Jagielski     osl_getProcessWorkingDir(&iniDir.pData);
394*b1cdbd2cSJim Jagielski 
395*b1cdbd2cSJim Jagielski     Reference<lang::XMultiComponentFactory> smgr_XMultiComponentFactory(
396*b1cdbd2cSJim Jagielski         bootstrapInitialSF(bootstrapPath) );
397*b1cdbd2cSJim Jagielski     Reference<lang::XMultiServiceFactory> smgr_XMultiServiceFactory(
398*b1cdbd2cSJim Jagielski         smgr_XMultiComponentFactory, UNO_QUERY );
399*b1cdbd2cSJim Jagielski 
400*b1cdbd2cSJim Jagielski     Reference<registry::XRegistryKey> xEmptyKey;
401*b1cdbd2cSJim Jagielski     Reference<lang::XSingleServiceFactory> xSimRegFac(
402*b1cdbd2cSJim Jagielski         loadSharedLibComponentFactory(
403*b1cdbd2cSJim Jagielski             OUSTR("bootstrap.uno" SAL_DLLEXTENSION), bootstrapPath,
404*b1cdbd2cSJim Jagielski             OUSTR("com.sun.star.comp.stoc.SimpleRegistry"),
405*b1cdbd2cSJim Jagielski             smgr_XMultiServiceFactory,
406*b1cdbd2cSJim Jagielski             xEmptyKey),
407*b1cdbd2cSJim Jagielski         UNO_QUERY);
408*b1cdbd2cSJim Jagielski 
409*b1cdbd2cSJim Jagielski     Reference<lang::XSingleServiceFactory> xNesRegFac(
410*b1cdbd2cSJim Jagielski         loadSharedLibComponentFactory(
411*b1cdbd2cSJim Jagielski             OUSTR("bootstrap.uno" SAL_DLLEXTENSION), bootstrapPath,
412*b1cdbd2cSJim Jagielski             OUSTR("com.sun.star.comp.stoc.NestedRegistry"),
413*b1cdbd2cSJim Jagielski             smgr_XMultiServiceFactory,
414*b1cdbd2cSJim Jagielski             xEmptyKey),
415*b1cdbd2cSJim Jagielski         UNO_QUERY);
416*b1cdbd2cSJim Jagielski 
417*b1cdbd2cSJim Jagielski     sal_Bool bFallenback_types;
418*b1cdbd2cSJim Jagielski     OUString cls_uno_types =
419*b1cdbd2cSJim Jagielski         findBoostrapArgument( bootstrap, OUSTR("TYPES"), &bFallenback_types );
420*b1cdbd2cSJim Jagielski 
421*b1cdbd2cSJim Jagielski     Reference<registry::XSimpleRegistry> types_xRegistry =
422*b1cdbd2cSJim Jagielski         nestRegistries(
423*b1cdbd2cSJim Jagielski             iniDir, xSimRegFac, xNesRegFac, cls_uno_types,
424*b1cdbd2cSJim Jagielski             OUString(), sal_False, bFallenback_types );
425*b1cdbd2cSJim Jagielski 
426*b1cdbd2cSJim Jagielski     // ==== bootstrap from services registry ====
427*b1cdbd2cSJim Jagielski 
428*b1cdbd2cSJim Jagielski     sal_Bool bFallenback_services;
429*b1cdbd2cSJim Jagielski     OUString cls_uno_services = findBoostrapArgument(
430*b1cdbd2cSJim Jagielski         bootstrap, OUSTR("SERVICES"), &bFallenback_services );
431*b1cdbd2cSJim Jagielski 
432*b1cdbd2cSJim Jagielski     sal_Bool fallenBackWriteRegistry;
433*b1cdbd2cSJim Jagielski     OUString write_rdb = findBoostrapArgument(
434*b1cdbd2cSJim Jagielski         bootstrap, OUSTR("WRITERDB"), &fallenBackWriteRegistry );
435*b1cdbd2cSJim Jagielski     if (fallenBackWriteRegistry)
436*b1cdbd2cSJim Jagielski     {
437*b1cdbd2cSJim Jagielski         // no standard write rdb anymore
438*b1cdbd2cSJim Jagielski         write_rdb = OUString();
439*b1cdbd2cSJim Jagielski     }
440*b1cdbd2cSJim Jagielski 
441*b1cdbd2cSJim Jagielski     Reference<registry::XSimpleRegistry> services_xRegistry = nestRegistries(
442*b1cdbd2cSJim Jagielski         iniDir, xSimRegFac, xNesRegFac, cls_uno_services, write_rdb,
443*b1cdbd2cSJim Jagielski         !fallenBackWriteRegistry, bFallenback_services );
444*b1cdbd2cSJim Jagielski 
445*b1cdbd2cSJim Jagielski     Reference< XComponentContext > xContext(
446*b1cdbd2cSJim Jagielski         bootstrapInitialContext(
447*b1cdbd2cSJim Jagielski             smgr_XMultiComponentFactory, types_xRegistry, services_xRegistry,
448*b1cdbd2cSJim Jagielski             bootstrapPath, bootstrap ) );
449*b1cdbd2cSJim Jagielski 
450*b1cdbd2cSJim Jagielski     // initialize sf
451*b1cdbd2cSJim Jagielski     Reference< lang::XInitialization > xInit(
452*b1cdbd2cSJim Jagielski         smgr_XMultiComponentFactory, UNO_QUERY );
453*b1cdbd2cSJim Jagielski     OSL_ASSERT( xInit.is() );
454*b1cdbd2cSJim Jagielski     Sequence< Any > aSFInit( 1 );
455*b1cdbd2cSJim Jagielski     aSFInit[ 0 ] <<= services_xRegistry;
456*b1cdbd2cSJim Jagielski     xInit->initialize( aSFInit );
457*b1cdbd2cSJim Jagielski 
458*b1cdbd2cSJim Jagielski     return xContext;
459*b1cdbd2cSJim Jagielski }
460*b1cdbd2cSJim Jagielski 
461*b1cdbd2cSJim Jagielski }
462*b1cdbd2cSJim Jagielski 
463*b1cdbd2cSJim Jagielski Reference< XComponentContext >
defaultBootstrap_InitialComponentContext(OUString const & iniFile)464*b1cdbd2cSJim Jagielski SAL_CALL defaultBootstrap_InitialComponentContext(
465*b1cdbd2cSJim Jagielski     OUString const & iniFile )
466*b1cdbd2cSJim Jagielski     SAL_THROW( (Exception) )
467*b1cdbd2cSJim Jagielski {
468*b1cdbd2cSJim Jagielski     Bootstrap bootstrap( iniFile );
469*b1cdbd2cSJim Jagielski     if (bootstrap.getHandle() == 0)
470*b1cdbd2cSJim Jagielski         throw io::IOException(OUSTR("Cannot open for reading: ") + iniFile, 0);
471*b1cdbd2cSJim Jagielski     return defaultBootstrap_InitialComponentContext( bootstrap );
472*b1cdbd2cSJim Jagielski }
473*b1cdbd2cSJim Jagielski 
474*b1cdbd2cSJim Jagielski Reference< XComponentContext >
defaultBootstrap_InitialComponentContext()475*b1cdbd2cSJim Jagielski SAL_CALL defaultBootstrap_InitialComponentContext()
476*b1cdbd2cSJim Jagielski     SAL_THROW( (Exception) )
477*b1cdbd2cSJim Jagielski {
478*b1cdbd2cSJim Jagielski     return defaultBootstrap_InitialComponentContext( get_unorc() );
479*b1cdbd2cSJim Jagielski }
480*b1cdbd2cSJim Jagielski 
BootstrapException()481*b1cdbd2cSJim Jagielski BootstrapException::BootstrapException()
482*b1cdbd2cSJim Jagielski {
483*b1cdbd2cSJim Jagielski }
484*b1cdbd2cSJim Jagielski 
BootstrapException(const::rtl::OUString & rMessage)485*b1cdbd2cSJim Jagielski BootstrapException::BootstrapException( const ::rtl::OUString & rMessage )
486*b1cdbd2cSJim Jagielski     :m_aMessage( rMessage )
487*b1cdbd2cSJim Jagielski {
488*b1cdbd2cSJim Jagielski }
489*b1cdbd2cSJim Jagielski 
BootstrapException(const BootstrapException & e)490*b1cdbd2cSJim Jagielski BootstrapException::BootstrapException( const BootstrapException & e )
491*b1cdbd2cSJim Jagielski {
492*b1cdbd2cSJim Jagielski     m_aMessage = e.m_aMessage;
493*b1cdbd2cSJim Jagielski }
494*b1cdbd2cSJim Jagielski 
~BootstrapException()495*b1cdbd2cSJim Jagielski BootstrapException::~BootstrapException()
496*b1cdbd2cSJim Jagielski {
497*b1cdbd2cSJim Jagielski }
498*b1cdbd2cSJim Jagielski 
operator =(const BootstrapException & e)499*b1cdbd2cSJim Jagielski BootstrapException & BootstrapException::operator=( const BootstrapException & e )
500*b1cdbd2cSJim Jagielski {
501*b1cdbd2cSJim Jagielski     m_aMessage = e.m_aMessage;
502*b1cdbd2cSJim Jagielski     return *this;
503*b1cdbd2cSJim Jagielski }
504*b1cdbd2cSJim Jagielski 
getMessage() const505*b1cdbd2cSJim Jagielski const ::rtl::OUString & BootstrapException::getMessage() const
506*b1cdbd2cSJim Jagielski {
507*b1cdbd2cSJim Jagielski     return m_aMessage;
508*b1cdbd2cSJim Jagielski }
509*b1cdbd2cSJim Jagielski 
bootstrap()510*b1cdbd2cSJim Jagielski Reference< XComponentContext > SAL_CALL bootstrap()
511*b1cdbd2cSJim Jagielski {
512*b1cdbd2cSJim Jagielski     Reference< XComponentContext > xRemoteContext;
513*b1cdbd2cSJim Jagielski 
514*b1cdbd2cSJim Jagielski     try
515*b1cdbd2cSJim Jagielski     {
516*b1cdbd2cSJim Jagielski         char const * p1 = cppuhelper_detail_findSofficePath();
517*b1cdbd2cSJim Jagielski         if (p1 == NULL) {
518*b1cdbd2cSJim Jagielski             throw BootstrapException(
519*b1cdbd2cSJim Jagielski                 OUSTR("no soffice installation found!"));
520*b1cdbd2cSJim Jagielski         }
521*b1cdbd2cSJim Jagielski         rtl::OUString p2;
522*b1cdbd2cSJim Jagielski         if (!rtl_convertStringToUString(
523*b1cdbd2cSJim Jagielski                 &p2.pData, p1, strlen(p1), osl_getThreadTextEncoding(),
524*b1cdbd2cSJim Jagielski                 (RTL_TEXTTOUNICODE_FLAGS_UNDEFINED_ERROR |
525*b1cdbd2cSJim Jagielski                  RTL_TEXTTOUNICODE_FLAGS_MBUNDEFINED_ERROR |
526*b1cdbd2cSJim Jagielski                  RTL_TEXTTOUNICODE_FLAGS_INVALID_ERROR)))
527*b1cdbd2cSJim Jagielski         {
528*b1cdbd2cSJim Jagielski             throw BootstrapException(
529*b1cdbd2cSJim Jagielski                 OUSTR("bad characters in soffice installation path!"));
530*b1cdbd2cSJim Jagielski         }
531*b1cdbd2cSJim Jagielski         OUString path;
532*b1cdbd2cSJim Jagielski         if (osl::FileBase::getFileURLFromSystemPath(p2, path) !=
533*b1cdbd2cSJim Jagielski             osl::FileBase::E_None)
534*b1cdbd2cSJim Jagielski         {
535*b1cdbd2cSJim Jagielski             throw BootstrapException(
536*b1cdbd2cSJim Jagielski                 OUSTR("cannot convert soffice installation path to URL!"));
537*b1cdbd2cSJim Jagielski         }
538*b1cdbd2cSJim Jagielski         if (path.getLength() > 0 && path[path.getLength() - 1] != '/') {
539*b1cdbd2cSJim Jagielski             path += OUSTR("/");
540*b1cdbd2cSJim Jagielski         }
541*b1cdbd2cSJim Jagielski 
542*b1cdbd2cSJim Jagielski         OUString uri;
543*b1cdbd2cSJim Jagielski         if (!Bootstrap::get(OUSTR("URE_BOOTSTRAP"), uri)) {
544*b1cdbd2cSJim Jagielski             Bootstrap::set(
545*b1cdbd2cSJim Jagielski                 OUSTR("URE_BOOTSTRAP"),
546*b1cdbd2cSJim Jagielski                 Bootstrap::encode(path + OUSTR(SAL_CONFIGFILE("fundamental"))));
547*b1cdbd2cSJim Jagielski         }
548*b1cdbd2cSJim Jagielski 
549*b1cdbd2cSJim Jagielski         // create default local component context
550*b1cdbd2cSJim Jagielski         Reference< XComponentContext > xLocalContext(
551*b1cdbd2cSJim Jagielski             defaultBootstrap_InitialComponentContext() );
552*b1cdbd2cSJim Jagielski         if ( !xLocalContext.is() )
553*b1cdbd2cSJim Jagielski             throw BootstrapException( OUSTR( "no local component context!" ) );
554*b1cdbd2cSJim Jagielski 
555*b1cdbd2cSJim Jagielski         // create a random pipe name
556*b1cdbd2cSJim Jagielski         rtlRandomPool hPool = rtl_random_createPool();
557*b1cdbd2cSJim Jagielski         if ( hPool == 0 )
558*b1cdbd2cSJim Jagielski             throw BootstrapException( OUSTR( "cannot create random pool!" ) );
559*b1cdbd2cSJim Jagielski         sal_uInt8 bytes[ 16 ];
560*b1cdbd2cSJim Jagielski         if ( rtl_random_getBytes( hPool, bytes, ARLEN( bytes ) )
561*b1cdbd2cSJim Jagielski             != rtl_Random_E_None )
562*b1cdbd2cSJim Jagielski             throw BootstrapException( OUSTR( "random pool error!" ) );
563*b1cdbd2cSJim Jagielski         rtl_random_destroyPool( hPool );
564*b1cdbd2cSJim Jagielski         ::rtl::OUStringBuffer buf;
565*b1cdbd2cSJim Jagielski         buf.appendAscii( RTL_CONSTASCII_STRINGPARAM( "uno" ) );
566*b1cdbd2cSJim Jagielski         for ( sal_uInt32 i = 0; i < ARLEN( bytes ); ++i )
567*b1cdbd2cSJim Jagielski             buf.append( static_cast< sal_Int32 >( bytes[ i ] ) );
568*b1cdbd2cSJim Jagielski         OUString sPipeName( buf.makeStringAndClear() );
569*b1cdbd2cSJim Jagielski 
570*b1cdbd2cSJim Jagielski         // accept string
571*b1cdbd2cSJim Jagielski         OSL_ASSERT( buf.getLength() == 0 );
572*b1cdbd2cSJim Jagielski         buf.appendAscii( RTL_CONSTASCII_STRINGPARAM( "-accept=pipe,name=" ) );
573*b1cdbd2cSJim Jagielski         buf.append( sPipeName );
574*b1cdbd2cSJim Jagielski         buf.appendAscii( RTL_CONSTASCII_STRINGPARAM( ";urp;" ) );
575*b1cdbd2cSJim Jagielski 
576*b1cdbd2cSJim Jagielski         // arguments
577*b1cdbd2cSJim Jagielski         OUString args [] = {
578*b1cdbd2cSJim Jagielski             OUSTR( "-nologo" ),
579*b1cdbd2cSJim Jagielski             OUSTR( "-nodefault" ),
580*b1cdbd2cSJim Jagielski             OUSTR( "-norestore" ),
581*b1cdbd2cSJim Jagielski             OUSTR( "-nocrashreport" ),
582*b1cdbd2cSJim Jagielski             OUSTR( "-nolockcheck" ),
583*b1cdbd2cSJim Jagielski             buf.makeStringAndClear()
584*b1cdbd2cSJim Jagielski         };
585*b1cdbd2cSJim Jagielski         rtl_uString * ar_args [] = {
586*b1cdbd2cSJim Jagielski             args[ 0 ].pData,
587*b1cdbd2cSJim Jagielski             args[ 1 ].pData,
588*b1cdbd2cSJim Jagielski             args[ 2 ].pData,
589*b1cdbd2cSJim Jagielski             args[ 3 ].pData,
590*b1cdbd2cSJim Jagielski             args[ 4 ].pData,
591*b1cdbd2cSJim Jagielski             args[ 5 ].pData
592*b1cdbd2cSJim Jagielski         };
593*b1cdbd2cSJim Jagielski         ::osl::Security sec;
594*b1cdbd2cSJim Jagielski 
595*b1cdbd2cSJim Jagielski         // start office process
596*b1cdbd2cSJim Jagielski         oslProcess hProcess = 0;
597*b1cdbd2cSJim Jagielski         oslProcessError rc = osl_executeProcess(
598*b1cdbd2cSJim Jagielski             (path + OUSTR("soffice")).pData, ar_args, ARLEN( ar_args ),
599*b1cdbd2cSJim Jagielski             osl_Process_DETACHED,
600*b1cdbd2cSJim Jagielski             sec.getHandle(),
601*b1cdbd2cSJim Jagielski             0, // => current working dir
602*b1cdbd2cSJim Jagielski             0, 0, // => no env vars
603*b1cdbd2cSJim Jagielski             &hProcess );
604*b1cdbd2cSJim Jagielski         switch ( rc )
605*b1cdbd2cSJim Jagielski         {
606*b1cdbd2cSJim Jagielski             case osl_Process_E_None:
607*b1cdbd2cSJim Jagielski                 osl_freeProcessHandle( hProcess );
608*b1cdbd2cSJim Jagielski                 break;
609*b1cdbd2cSJim Jagielski             case osl_Process_E_NotFound:
610*b1cdbd2cSJim Jagielski                 throw BootstrapException( OUSTR( "image not found!" ) );
611*b1cdbd2cSJim Jagielski             case osl_Process_E_TimedOut:
612*b1cdbd2cSJim Jagielski                 throw BootstrapException( OUSTR( "timout occured!" ) );
613*b1cdbd2cSJim Jagielski             case osl_Process_E_NoPermission:
614*b1cdbd2cSJim Jagielski                 throw BootstrapException( OUSTR( "permission denied!" ) );
615*b1cdbd2cSJim Jagielski             case osl_Process_E_Unknown:
616*b1cdbd2cSJim Jagielski                 throw BootstrapException( OUSTR( "unknown error!" ) );
617*b1cdbd2cSJim Jagielski             case osl_Process_E_InvalidError:
618*b1cdbd2cSJim Jagielski             default:
619*b1cdbd2cSJim Jagielski                 throw BootstrapException( OUSTR( "unmapped error!" ) );
620*b1cdbd2cSJim Jagielski         }
621*b1cdbd2cSJim Jagielski 
622*b1cdbd2cSJim Jagielski         // create a URL resolver
623*b1cdbd2cSJim Jagielski         Reference< bridge::XUnoUrlResolver > xUrlResolver(
624*b1cdbd2cSJim Jagielski             bridge::UnoUrlResolver::create( xLocalContext ) );
625*b1cdbd2cSJim Jagielski 
626*b1cdbd2cSJim Jagielski         // connection string
627*b1cdbd2cSJim Jagielski         OSL_ASSERT( buf.getLength() == 0 );
628*b1cdbd2cSJim Jagielski         buf.appendAscii( RTL_CONSTASCII_STRINGPARAM( "uno:pipe,name=" ) );
629*b1cdbd2cSJim Jagielski         buf.append( sPipeName );
630*b1cdbd2cSJim Jagielski         buf.appendAscii( RTL_CONSTASCII_STRINGPARAM(
631*b1cdbd2cSJim Jagielski             ";urp;StarOffice.ComponentContext" ) );
632*b1cdbd2cSJim Jagielski         OUString sConnectString( buf.makeStringAndClear() );
633*b1cdbd2cSJim Jagielski 
634*b1cdbd2cSJim Jagielski         // wait until office is started
635*b1cdbd2cSJim Jagielski         for ( ; ; )
636*b1cdbd2cSJim Jagielski         {
637*b1cdbd2cSJim Jagielski             try
638*b1cdbd2cSJim Jagielski             {
639*b1cdbd2cSJim Jagielski                 // try to connect to office
640*b1cdbd2cSJim Jagielski                 xRemoteContext.set(
641*b1cdbd2cSJim Jagielski                     xUrlResolver->resolve( sConnectString ), UNO_QUERY_THROW );
642*b1cdbd2cSJim Jagielski                 break;
643*b1cdbd2cSJim Jagielski             }
644*b1cdbd2cSJim Jagielski             catch ( connection::NoConnectException & )
645*b1cdbd2cSJim Jagielski             {
646*b1cdbd2cSJim Jagielski                 // wait 500 ms, then try to connect again
647*b1cdbd2cSJim Jagielski                 TimeValue tv = { 0 /* secs */, 500000000 /* nanosecs */ };
648*b1cdbd2cSJim Jagielski                 ::osl::Thread::wait( tv );
649*b1cdbd2cSJim Jagielski             }
650*b1cdbd2cSJim Jagielski         }
651*b1cdbd2cSJim Jagielski     }
652*b1cdbd2cSJim Jagielski     catch ( Exception & e )
653*b1cdbd2cSJim Jagielski     {
654*b1cdbd2cSJim Jagielski         throw BootstrapException(
655*b1cdbd2cSJim Jagielski             OUSTR( "unexpected UNO exception caught: " ) + e.Message );
656*b1cdbd2cSJim Jagielski     }
657*b1cdbd2cSJim Jagielski 
658*b1cdbd2cSJim Jagielski     return xRemoteContext;
659*b1cdbd2cSJim Jagielski }
660*b1cdbd2cSJim Jagielski 
bootstrap_expandUri(OUString const & uri)661*b1cdbd2cSJim Jagielski OUString bootstrap_expandUri(OUString const & uri) {
662*b1cdbd2cSJim Jagielski     static char const PREFIX[] = "vnd.sun.star.expand:";
663*b1cdbd2cSJim Jagielski     return uri.matchAsciiL(RTL_CONSTASCII_STRINGPARAM(PREFIX))
664*b1cdbd2cSJim Jagielski         ? cppuhelper::detail::expandMacros(
665*b1cdbd2cSJim Jagielski             rtl::Uri::decode(
666*b1cdbd2cSJim Jagielski                 uri.copy(RTL_CONSTASCII_LENGTH(PREFIX)),
667*b1cdbd2cSJim Jagielski                 rtl_UriDecodeWithCharset, RTL_TEXTENCODING_UTF8))
668*b1cdbd2cSJim Jagielski         : uri;
669*b1cdbd2cSJim Jagielski }
670*b1cdbd2cSJim Jagielski 
671*b1cdbd2cSJim Jagielski } // namespace cppu
672