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_cppu.hxx"
26*b1cdbd2cSJim Jagielski 
27*b1cdbd2cSJim Jagielski #include "osl/thread.hxx"
28*b1cdbd2cSJim Jagielski #include "osl/conditn.hxx"
29*b1cdbd2cSJim Jagielski #include "osl/mutex.hxx"
30*b1cdbd2cSJim Jagielski 
31*b1cdbd2cSJim Jagielski #include "cppu/helper/purpenv/Environment.hxx"
32*b1cdbd2cSJim Jagielski #include "cppu/helper/purpenv/Mapping.hxx"
33*b1cdbd2cSJim Jagielski 
34*b1cdbd2cSJim Jagielski 
35*b1cdbd2cSJim Jagielski #ifdef debug
36*b1cdbd2cSJim Jagielski # define LOG_LIFECYCLE_AffineBridge
37*b1cdbd2cSJim Jagielski #endif
38*b1cdbd2cSJim Jagielski 
39*b1cdbd2cSJim Jagielski #ifdef LOG_LIFECYCLE_AffineBridge
40*b1cdbd2cSJim Jagielski #  include <iostream>
41*b1cdbd2cSJim Jagielski #  define LOG_LIFECYCLE_AffineBridge_emit(x) x
42*b1cdbd2cSJim Jagielski 
43*b1cdbd2cSJim Jagielski #else
44*b1cdbd2cSJim Jagielski #  define LOG_LIFECYCLE_AffineBridge_emit(x)
45*b1cdbd2cSJim Jagielski 
46*b1cdbd2cSJim Jagielski #endif
47*b1cdbd2cSJim Jagielski 
48*b1cdbd2cSJim Jagielski class InnerThread;
49*b1cdbd2cSJim Jagielski class OuterThread;
50*b1cdbd2cSJim Jagielski 
51*b1cdbd2cSJim Jagielski class SAL_DLLPRIVATE AffineBridge : public cppu::Enterable
52*b1cdbd2cSJim Jagielski {
53*b1cdbd2cSJim Jagielski public:
54*b1cdbd2cSJim Jagielski 	enum Msg
55*b1cdbd2cSJim Jagielski 	{
56*b1cdbd2cSJim Jagielski 		CB_DONE,
57*b1cdbd2cSJim Jagielski 		CB_FPOINTER
58*b1cdbd2cSJim Jagielski 	};
59*b1cdbd2cSJim Jagielski 
60*b1cdbd2cSJim Jagielski 	Msg                   m_message;
61*b1cdbd2cSJim Jagielski 	uno_EnvCallee       * m_pCallee;
62*b1cdbd2cSJim Jagielski 	va_list             * m_pParam;
63*b1cdbd2cSJim Jagielski 
64*b1cdbd2cSJim Jagielski 	osl::Mutex            m_innerMutex;
65*b1cdbd2cSJim Jagielski 	oslThreadIdentifier   m_innerThreadId;
66*b1cdbd2cSJim Jagielski 	InnerThread         * m_pInnerThread;
67*b1cdbd2cSJim Jagielski 	osl::Condition        m_innerCondition;
68*b1cdbd2cSJim Jagielski 	sal_Int32             m_enterCount;
69*b1cdbd2cSJim Jagielski 
70*b1cdbd2cSJim Jagielski 	osl::Mutex            m_outerMutex;
71*b1cdbd2cSJim Jagielski 	oslThreadIdentifier   m_outerThreadId;
72*b1cdbd2cSJim Jagielski 	osl::Condition        m_outerCondition;
73*b1cdbd2cSJim Jagielski 	OuterThread         * m_pOuterThread;
74*b1cdbd2cSJim Jagielski 
75*b1cdbd2cSJim Jagielski 	explicit  AffineBridge(void);
76*b1cdbd2cSJim Jagielski 	virtual  ~AffineBridge(void);
77*b1cdbd2cSJim Jagielski 
78*b1cdbd2cSJim Jagielski 	virtual void  v_callInto_v(uno_EnvCallee * pCallee, va_list * pParam);
79*b1cdbd2cSJim Jagielski 	virtual void  v_callOut_v (uno_EnvCallee * pCallee, va_list * pParam);
80*b1cdbd2cSJim Jagielski 
81*b1cdbd2cSJim Jagielski 	virtual void  v_enter(void);
82*b1cdbd2cSJim Jagielski 	virtual void  v_leave(void);
83*b1cdbd2cSJim Jagielski 
84*b1cdbd2cSJim Jagielski 	virtual int  v_isValid(rtl::OUString * pReason);
85*b1cdbd2cSJim Jagielski 
86*b1cdbd2cSJim Jagielski 	void innerDispatch(void);
87*b1cdbd2cSJim Jagielski 	void outerDispatch(int loop);
88*b1cdbd2cSJim Jagielski };
89*b1cdbd2cSJim Jagielski 
90*b1cdbd2cSJim Jagielski class SAL_DLLPRIVATE InnerThread : public osl::Thread
91*b1cdbd2cSJim Jagielski {
92*b1cdbd2cSJim Jagielski     virtual void SAL_CALL run(void);
93*b1cdbd2cSJim Jagielski 
94*b1cdbd2cSJim Jagielski 	AffineBridge * m_pAffineBridge;
95*b1cdbd2cSJim Jagielski 
96*b1cdbd2cSJim Jagielski public:
InnerThread(AffineBridge * threadEnvironment)97*b1cdbd2cSJim Jagielski 	InnerThread(AffineBridge * threadEnvironment)
98*b1cdbd2cSJim Jagielski 		: m_pAffineBridge(threadEnvironment)
99*b1cdbd2cSJim Jagielski 		{
100*b1cdbd2cSJim Jagielski 			create();
101*b1cdbd2cSJim Jagielski 		}
102*b1cdbd2cSJim Jagielski };
103*b1cdbd2cSJim Jagielski 
run(void)104*b1cdbd2cSJim Jagielski void InnerThread::run(void)
105*b1cdbd2cSJim Jagielski {
106*b1cdbd2cSJim Jagielski 	m_pAffineBridge->enter();
107*b1cdbd2cSJim Jagielski 	m_pAffineBridge->innerDispatch();
108*b1cdbd2cSJim Jagielski 	m_pAffineBridge->leave();
109*b1cdbd2cSJim Jagielski }
110*b1cdbd2cSJim Jagielski 
111*b1cdbd2cSJim Jagielski class SAL_DLLPRIVATE OuterThread : public osl::Thread
112*b1cdbd2cSJim Jagielski {
113*b1cdbd2cSJim Jagielski     virtual void SAL_CALL run(void);
114*b1cdbd2cSJim Jagielski 
115*b1cdbd2cSJim Jagielski 	AffineBridge * m_pAffineBridge;
116*b1cdbd2cSJim Jagielski 
117*b1cdbd2cSJim Jagielski public:
118*b1cdbd2cSJim Jagielski 	OuterThread(AffineBridge * threadEnvironment);
119*b1cdbd2cSJim Jagielski };
120*b1cdbd2cSJim Jagielski 
OuterThread(AffineBridge * threadEnvironment)121*b1cdbd2cSJim Jagielski OuterThread::OuterThread(AffineBridge * threadEnvironment)
122*b1cdbd2cSJim Jagielski 	: m_pAffineBridge(threadEnvironment)
123*b1cdbd2cSJim Jagielski {
124*b1cdbd2cSJim Jagielski 	create();
125*b1cdbd2cSJim Jagielski }
126*b1cdbd2cSJim Jagielski 
run(void)127*b1cdbd2cSJim Jagielski void OuterThread::run(void)
128*b1cdbd2cSJim Jagielski {
129*b1cdbd2cSJim Jagielski 	osl::MutexGuard guard(m_pAffineBridge->m_outerMutex);
130*b1cdbd2cSJim Jagielski 
131*b1cdbd2cSJim Jagielski 	m_pAffineBridge->m_outerThreadId = getIdentifier();
132*b1cdbd2cSJim Jagielski 	m_pAffineBridge->outerDispatch(0);
133*b1cdbd2cSJim Jagielski 	m_pAffineBridge->m_outerThreadId = 0;
134*b1cdbd2cSJim Jagielski 
135*b1cdbd2cSJim Jagielski 	m_pAffineBridge->m_pOuterThread = NULL;
136*b1cdbd2cSJim Jagielski 	m_pAffineBridge = NULL;
137*b1cdbd2cSJim Jagielski }
138*b1cdbd2cSJim Jagielski 
139*b1cdbd2cSJim Jagielski 
AffineBridge(void)140*b1cdbd2cSJim Jagielski AffineBridge::AffineBridge(void)
141*b1cdbd2cSJim Jagielski 	: m_innerThreadId(0),
142*b1cdbd2cSJim Jagielski 	  m_pInnerThread (NULL),
143*b1cdbd2cSJim Jagielski 	  m_enterCount   (0),
144*b1cdbd2cSJim Jagielski 	  m_outerThreadId(0),
145*b1cdbd2cSJim Jagielski 	  m_pOuterThread (NULL)
146*b1cdbd2cSJim Jagielski {
147*b1cdbd2cSJim Jagielski 	LOG_LIFECYCLE_AffineBridge_emit(fprintf(stderr, "LIFE: %s -> %p\n", "AffineBridge::AffineBridge(uno_Environment * pEnv)", this));
148*b1cdbd2cSJim Jagielski }
149*b1cdbd2cSJim Jagielski 
~AffineBridge(void)150*b1cdbd2cSJim Jagielski AffineBridge::~AffineBridge(void)
151*b1cdbd2cSJim Jagielski {
152*b1cdbd2cSJim Jagielski 	LOG_LIFECYCLE_AffineBridge_emit(fprintf(stderr, "LIFE: %s -> %p\n", "AffineBridge::~AffineBridge(void)", this));
153*b1cdbd2cSJim Jagielski 
154*b1cdbd2cSJim Jagielski 	if (m_pInnerThread && osl_getThreadIdentifier(NULL) != m_innerThreadId)
155*b1cdbd2cSJim Jagielski 	{
156*b1cdbd2cSJim Jagielski 		m_message = CB_DONE;
157*b1cdbd2cSJim Jagielski 		m_innerCondition.set();
158*b1cdbd2cSJim Jagielski 
159*b1cdbd2cSJim Jagielski 		m_pInnerThread->join();
160*b1cdbd2cSJim Jagielski 	}
161*b1cdbd2cSJim Jagielski 
162*b1cdbd2cSJim Jagielski 	delete m_pInnerThread;
163*b1cdbd2cSJim Jagielski 
164*b1cdbd2cSJim Jagielski 	if (m_pOuterThread)
165*b1cdbd2cSJim Jagielski 	{
166*b1cdbd2cSJim Jagielski 		m_pOuterThread->join();
167*b1cdbd2cSJim Jagielski 		delete m_pOuterThread;
168*b1cdbd2cSJim Jagielski 	}
169*b1cdbd2cSJim Jagielski }
170*b1cdbd2cSJim Jagielski 
171*b1cdbd2cSJim Jagielski 
outerDispatch(int loop)172*b1cdbd2cSJim Jagielski void AffineBridge::outerDispatch(int loop)
173*b1cdbd2cSJim Jagielski {
174*b1cdbd2cSJim Jagielski 	OSL_ASSERT(m_outerThreadId == osl_getThreadIdentifier(NULL));
175*b1cdbd2cSJim Jagielski 	OSL_ASSERT(m_innerThreadId != m_outerThreadId);
176*b1cdbd2cSJim Jagielski 
177*b1cdbd2cSJim Jagielski 	Msg mm;
178*b1cdbd2cSJim Jagielski 
179*b1cdbd2cSJim Jagielski 	do
180*b1cdbd2cSJim Jagielski 	{
181*b1cdbd2cSJim Jagielski 		// FIXME: created outer thread must not wait
182*b1cdbd2cSJim Jagielski 		// in case of no message
183*b1cdbd2cSJim Jagielski 		// note: no message can happen in case newly created
184*b1cdbd2cSJim Jagielski 		// outer thread acquire outerMutex after a real outer
185*b1cdbd2cSJim Jagielski 		// thread enters outerDispatch!
186*b1cdbd2cSJim Jagielski 		m_outerCondition.wait();
187*b1cdbd2cSJim Jagielski 		m_outerCondition.reset();
188*b1cdbd2cSJim Jagielski 
189*b1cdbd2cSJim Jagielski 		mm = m_message;
190*b1cdbd2cSJim Jagielski 
191*b1cdbd2cSJim Jagielski 		switch(mm)
192*b1cdbd2cSJim Jagielski 		{
193*b1cdbd2cSJim Jagielski 		case CB_DONE:
194*b1cdbd2cSJim Jagielski 			break;
195*b1cdbd2cSJim Jagielski 
196*b1cdbd2cSJim Jagielski 		case CB_FPOINTER:
197*b1cdbd2cSJim Jagielski 		{
198*b1cdbd2cSJim Jagielski 			m_pCallee(m_pParam);
199*b1cdbd2cSJim Jagielski 
200*b1cdbd2cSJim Jagielski 			m_message = CB_DONE;
201*b1cdbd2cSJim Jagielski 			m_innerCondition.set();
202*b1cdbd2cSJim Jagielski 			break;
203*b1cdbd2cSJim Jagielski 		}
204*b1cdbd2cSJim Jagielski 		default:
205*b1cdbd2cSJim Jagielski 			abort();
206*b1cdbd2cSJim Jagielski 		}
207*b1cdbd2cSJim Jagielski 	}
208*b1cdbd2cSJim Jagielski 	while(mm != CB_DONE && loop);
209*b1cdbd2cSJim Jagielski }
210*b1cdbd2cSJim Jagielski 
innerDispatch(void)211*b1cdbd2cSJim Jagielski void AffineBridge::innerDispatch(void)
212*b1cdbd2cSJim Jagielski {
213*b1cdbd2cSJim Jagielski 	OSL_ASSERT(m_innerThreadId == osl_getThreadIdentifier(NULL));
214*b1cdbd2cSJim Jagielski 	OSL_ASSERT(m_innerThreadId != m_outerThreadId);
215*b1cdbd2cSJim Jagielski 
216*b1cdbd2cSJim Jagielski 	Msg mm;
217*b1cdbd2cSJim Jagielski 
218*b1cdbd2cSJim Jagielski 	do
219*b1cdbd2cSJim Jagielski 	{
220*b1cdbd2cSJim Jagielski 		m_innerCondition.wait();
221*b1cdbd2cSJim Jagielski 		m_innerCondition.reset();
222*b1cdbd2cSJim Jagielski 
223*b1cdbd2cSJim Jagielski 		mm = m_message;
224*b1cdbd2cSJim Jagielski 
225*b1cdbd2cSJim Jagielski 		switch(mm)
226*b1cdbd2cSJim Jagielski 		{
227*b1cdbd2cSJim Jagielski 		case CB_DONE:
228*b1cdbd2cSJim Jagielski 			break;
229*b1cdbd2cSJim Jagielski 
230*b1cdbd2cSJim Jagielski 		case CB_FPOINTER:
231*b1cdbd2cSJim Jagielski 		{
232*b1cdbd2cSJim Jagielski 			m_pCallee(m_pParam);
233*b1cdbd2cSJim Jagielski 
234*b1cdbd2cSJim Jagielski 			m_message = CB_DONE;
235*b1cdbd2cSJim Jagielski 			m_outerCondition.set();
236*b1cdbd2cSJim Jagielski 			break;
237*b1cdbd2cSJim Jagielski 		}
238*b1cdbd2cSJim Jagielski 		default:
239*b1cdbd2cSJim Jagielski 			abort();
240*b1cdbd2cSJim Jagielski 		}
241*b1cdbd2cSJim Jagielski 	}
242*b1cdbd2cSJim Jagielski 	while(mm != CB_DONE);
243*b1cdbd2cSJim Jagielski }
244*b1cdbd2cSJim Jagielski 
v_callInto_v(uno_EnvCallee * pCallee,va_list * pParam)245*b1cdbd2cSJim Jagielski void AffineBridge::v_callInto_v(uno_EnvCallee * pCallee, va_list * pParam)
246*b1cdbd2cSJim Jagielski {
247*b1cdbd2cSJim Jagielski 	osl::MutexGuard guard(m_outerMutex); // only one thread at a time can call into
248*b1cdbd2cSJim Jagielski 
249*b1cdbd2cSJim Jagielski 	if (m_innerThreadId == 0) // no inner thread yet
250*b1cdbd2cSJim Jagielski 	{
251*b1cdbd2cSJim Jagielski 		m_pInnerThread  = new InnerThread(this);
252*b1cdbd2cSJim Jagielski 		m_pInnerThread->resume();
253*b1cdbd2cSJim Jagielski 	}
254*b1cdbd2cSJim Jagielski 
255*b1cdbd2cSJim Jagielski 	bool resetId = false;
256*b1cdbd2cSJim Jagielski 	if (!m_outerThreadId)
257*b1cdbd2cSJim Jagielski 	{
258*b1cdbd2cSJim Jagielski 		m_outerThreadId = osl_getThreadIdentifier(NULL);
259*b1cdbd2cSJim Jagielski 		resetId = true;
260*b1cdbd2cSJim Jagielski 	}
261*b1cdbd2cSJim Jagielski 
262*b1cdbd2cSJim Jagielski 	m_message = CB_FPOINTER;
263*b1cdbd2cSJim Jagielski 	m_pCallee = pCallee;
264*b1cdbd2cSJim Jagielski 	m_pParam  = pParam;
265*b1cdbd2cSJim Jagielski 	m_innerCondition.set();
266*b1cdbd2cSJim Jagielski 
267*b1cdbd2cSJim Jagielski 	outerDispatch(1);
268*b1cdbd2cSJim Jagielski 
269*b1cdbd2cSJim Jagielski 	if (resetId)
270*b1cdbd2cSJim Jagielski 		m_outerThreadId = 0;
271*b1cdbd2cSJim Jagielski }
272*b1cdbd2cSJim Jagielski 
v_callOut_v(uno_EnvCallee * pCallee,va_list * pParam)273*b1cdbd2cSJim Jagielski void AffineBridge::v_callOut_v(uno_EnvCallee * pCallee, va_list * pParam)
274*b1cdbd2cSJim Jagielski {
275*b1cdbd2cSJim Jagielski 	OSL_ASSERT(m_innerThreadId);
276*b1cdbd2cSJim Jagielski 
277*b1cdbd2cSJim Jagielski 	osl::MutexGuard guard(m_innerMutex);
278*b1cdbd2cSJim Jagielski 
279*b1cdbd2cSJim Jagielski 	if (m_outerThreadId == 0) // no outer thread yet
280*b1cdbd2cSJim Jagielski 	{
281*b1cdbd2cSJim Jagielski 		osl::MutexGuard guard_m_outerMutex(m_outerMutex);
282*b1cdbd2cSJim Jagielski 
283*b1cdbd2cSJim Jagielski 		if (m_outerThreadId == 0)
284*b1cdbd2cSJim Jagielski 		{
285*b1cdbd2cSJim Jagielski 			if (m_pOuterThread)
286*b1cdbd2cSJim Jagielski 			{
287*b1cdbd2cSJim Jagielski 				m_pOuterThread->join();
288*b1cdbd2cSJim Jagielski 				delete m_pOuterThread;
289*b1cdbd2cSJim Jagielski 			}
290*b1cdbd2cSJim Jagielski 
291*b1cdbd2cSJim Jagielski 			m_pOuterThread = new OuterThread(this);
292*b1cdbd2cSJim Jagielski 		}
293*b1cdbd2cSJim Jagielski 	}
294*b1cdbd2cSJim Jagielski 
295*b1cdbd2cSJim Jagielski 	m_message = CB_FPOINTER;
296*b1cdbd2cSJim Jagielski 	m_pCallee = pCallee;
297*b1cdbd2cSJim Jagielski 	m_pParam  = pParam;
298*b1cdbd2cSJim Jagielski 	m_outerCondition.set();
299*b1cdbd2cSJim Jagielski 
300*b1cdbd2cSJim Jagielski 	innerDispatch();
301*b1cdbd2cSJim Jagielski }
302*b1cdbd2cSJim Jagielski 
v_enter(void)303*b1cdbd2cSJim Jagielski void AffineBridge::v_enter(void)
304*b1cdbd2cSJim Jagielski {
305*b1cdbd2cSJim Jagielski 	m_innerMutex.acquire();
306*b1cdbd2cSJim Jagielski 
307*b1cdbd2cSJim Jagielski 	if (!m_enterCount)
308*b1cdbd2cSJim Jagielski 		m_innerThreadId = osl_getThreadIdentifier(NULL);
309*b1cdbd2cSJim Jagielski 
310*b1cdbd2cSJim Jagielski 	OSL_ASSERT(m_innerThreadId == osl_getThreadIdentifier(NULL));
311*b1cdbd2cSJim Jagielski 
312*b1cdbd2cSJim Jagielski 	++ m_enterCount;
313*b1cdbd2cSJim Jagielski }
314*b1cdbd2cSJim Jagielski 
v_leave(void)315*b1cdbd2cSJim Jagielski void AffineBridge::v_leave(void)
316*b1cdbd2cSJim Jagielski {
317*b1cdbd2cSJim Jagielski 	OSL_ASSERT(m_innerThreadId == osl_getThreadIdentifier(NULL));
318*b1cdbd2cSJim Jagielski 
319*b1cdbd2cSJim Jagielski 	-- m_enterCount;
320*b1cdbd2cSJim Jagielski 	if (!m_enterCount)
321*b1cdbd2cSJim Jagielski 		m_innerThreadId = 0;
322*b1cdbd2cSJim Jagielski 
323*b1cdbd2cSJim Jagielski 	m_innerMutex.release();
324*b1cdbd2cSJim Jagielski }
325*b1cdbd2cSJim Jagielski 
v_isValid(rtl::OUString * pReason)326*b1cdbd2cSJim Jagielski int  AffineBridge::v_isValid(rtl::OUString * pReason)
327*b1cdbd2cSJim Jagielski {
328*b1cdbd2cSJim Jagielski 	int result = 1;
329*b1cdbd2cSJim Jagielski 
330*b1cdbd2cSJim Jagielski 	result = m_enterCount > 0;
331*b1cdbd2cSJim Jagielski 	if (!result)
332*b1cdbd2cSJim Jagielski 		*pReason = rtl::OUString(RTL_CONSTASCII_USTRINGPARAM("not entered"));
333*b1cdbd2cSJim Jagielski 
334*b1cdbd2cSJim Jagielski 	else
335*b1cdbd2cSJim Jagielski 	{
336*b1cdbd2cSJim Jagielski 		result = m_innerThreadId == osl_getThreadIdentifier(NULL);
337*b1cdbd2cSJim Jagielski 
338*b1cdbd2cSJim Jagielski 		if (!result)
339*b1cdbd2cSJim Jagielski 			*pReason = rtl::OUString(RTL_CONSTASCII_USTRINGPARAM("wrong thread"));
340*b1cdbd2cSJim Jagielski 	}
341*b1cdbd2cSJim Jagielski 
342*b1cdbd2cSJim Jagielski 	if (result)
343*b1cdbd2cSJim Jagielski 		*pReason = rtl::OUString(RTL_CONSTASCII_USTRINGPARAM("OK"));
344*b1cdbd2cSJim Jagielski 
345*b1cdbd2cSJim Jagielski 	return result;
346*b1cdbd2cSJim Jagielski }
347*b1cdbd2cSJim Jagielski 
uno_initEnvironment(uno_Environment * pEnv)348*b1cdbd2cSJim Jagielski extern "C" void SAL_DLLPUBLIC_EXPORT SAL_CALL uno_initEnvironment(uno_Environment * pEnv)
349*b1cdbd2cSJim Jagielski 	SAL_THROW_EXTERN_C()
350*b1cdbd2cSJim Jagielski {
351*b1cdbd2cSJim Jagielski     cppu::helper::purpenv::Environment_initWithEnterable(pEnv, new AffineBridge());
352*b1cdbd2cSJim Jagielski }
353*b1cdbd2cSJim Jagielski 
uno_ext_getMapping(uno_Mapping ** ppMapping,uno_Environment * pFrom,uno_Environment * pTo)354*b1cdbd2cSJim Jagielski extern "C" void SAL_DLLPUBLIC_EXPORT SAL_CALL uno_ext_getMapping(uno_Mapping     ** ppMapping,
355*b1cdbd2cSJim Jagielski 														uno_Environment  * pFrom,
356*b1cdbd2cSJim Jagielski 														uno_Environment  * pTo )
357*b1cdbd2cSJim Jagielski {
358*b1cdbd2cSJim Jagielski 	cppu::helper::purpenv::createMapping(ppMapping, pFrom, pTo);
359*b1cdbd2cSJim Jagielski }
360*b1cdbd2cSJim Jagielski 
361