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