xref: /aoo4110/main/binaryurp/source/unmarshal.cxx (revision b1cdbd2c)
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 #include "sal/config.h"
25*b1cdbd2cSJim Jagielski 
26*b1cdbd2cSJim Jagielski #include <cstdlib>
27*b1cdbd2cSJim Jagielski #include <new>
28*b1cdbd2cSJim Jagielski #include <vector>
29*b1cdbd2cSJim Jagielski 
30*b1cdbd2cSJim Jagielski #include "boost/noncopyable.hpp"
31*b1cdbd2cSJim Jagielski #include "com/sun/star/io/IOException.hpp"
32*b1cdbd2cSJim Jagielski #include "com/sun/star/uno/Reference.hxx"
33*b1cdbd2cSJim Jagielski #include "com/sun/star/uno/RuntimeException.hpp"
34*b1cdbd2cSJim Jagielski #include "com/sun/star/uno/Sequence.hxx"
35*b1cdbd2cSJim Jagielski #include "com/sun/star/uno/XInterface.hpp"
36*b1cdbd2cSJim Jagielski #include "cppu/unotype.hxx"
37*b1cdbd2cSJim Jagielski #include "osl/diagnose.h"
38*b1cdbd2cSJim Jagielski #include "rtl/byteseq.hxx"
39*b1cdbd2cSJim Jagielski #include "rtl/ref.hxx"
40*b1cdbd2cSJim Jagielski #include "rtl/textcvt.h"
41*b1cdbd2cSJim Jagielski #include "rtl/textenc.h"
42*b1cdbd2cSJim Jagielski #include "rtl/ustring.h"
43*b1cdbd2cSJim Jagielski #include "rtl/ustring.hxx"
44*b1cdbd2cSJim Jagielski #include "sal/types.h"
45*b1cdbd2cSJim Jagielski #include "typelib/typeclass.h"
46*b1cdbd2cSJim Jagielski #include "typelib/typedescription.h"
47*b1cdbd2cSJim Jagielski #include "typelib/typedescription.hxx"
48*b1cdbd2cSJim Jagielski #include "uno/any2.h"
49*b1cdbd2cSJim Jagielski #include "uno/data.h"
50*b1cdbd2cSJim Jagielski #include "uno/dispatcher.hxx"
51*b1cdbd2cSJim Jagielski 
52*b1cdbd2cSJim Jagielski #include "binaryany.hxx"
53*b1cdbd2cSJim Jagielski #include "bridge.hxx"
54*b1cdbd2cSJim Jagielski #include "cache.hxx"
55*b1cdbd2cSJim Jagielski #include "readerstate.hxx"
56*b1cdbd2cSJim Jagielski #include "unmarshal.hxx"
57*b1cdbd2cSJim Jagielski 
58*b1cdbd2cSJim Jagielski namespace binaryurp {
59*b1cdbd2cSJim Jagielski 
60*b1cdbd2cSJim Jagielski namespace {
61*b1cdbd2cSJim Jagielski 
62*b1cdbd2cSJim Jagielski namespace css = com::sun::star;
63*b1cdbd2cSJim Jagielski 
allocate(sal_Size size)64*b1cdbd2cSJim Jagielski void * allocate(sal_Size size) {
65*b1cdbd2cSJim Jagielski     void * p = rtl_allocateMemory(size);
66*b1cdbd2cSJim Jagielski     if (p == 0) {
67*b1cdbd2cSJim Jagielski         throw std::bad_alloc();
68*b1cdbd2cSJim Jagielski     }
69*b1cdbd2cSJim Jagielski     return p;
70*b1cdbd2cSJim Jagielski }
71*b1cdbd2cSJim Jagielski 
copyMemberValues(css::uno::TypeDescription const & type,std::vector<BinaryAny>::iterator const & it,void * buffer)72*b1cdbd2cSJim Jagielski std::vector< BinaryAny >::iterator copyMemberValues(
73*b1cdbd2cSJim Jagielski     css::uno::TypeDescription const & type,
74*b1cdbd2cSJim Jagielski     std::vector< BinaryAny >::iterator const & it, void * buffer) throw ()
75*b1cdbd2cSJim Jagielski {
76*b1cdbd2cSJim Jagielski     OSL_ASSERT(
77*b1cdbd2cSJim Jagielski         type.is() &&
78*b1cdbd2cSJim Jagielski         (type.get()->eTypeClass == typelib_TypeClass_STRUCT ||
79*b1cdbd2cSJim Jagielski          type.get()->eTypeClass == typelib_TypeClass_EXCEPTION) &&
80*b1cdbd2cSJim Jagielski         buffer != 0);
81*b1cdbd2cSJim Jagielski     type.makeComplete();
82*b1cdbd2cSJim Jagielski     std::vector< BinaryAny >::iterator i(it);
83*b1cdbd2cSJim Jagielski     typelib_CompoundTypeDescription * ctd =
84*b1cdbd2cSJim Jagielski         reinterpret_cast< typelib_CompoundTypeDescription * >(type.get());
85*b1cdbd2cSJim Jagielski     if (ctd->pBaseTypeDescription != 0) {
86*b1cdbd2cSJim Jagielski         i = copyMemberValues(
87*b1cdbd2cSJim Jagielski             css::uno::TypeDescription(&ctd->pBaseTypeDescription->aBase), i,
88*b1cdbd2cSJim Jagielski             buffer);
89*b1cdbd2cSJim Jagielski     }
90*b1cdbd2cSJim Jagielski     for (sal_Int32 j = 0; j != ctd->nMembers; ++j) {
91*b1cdbd2cSJim Jagielski         uno_type_copyData(
92*b1cdbd2cSJim Jagielski             static_cast< char * >(buffer) + ctd->pMemberOffsets[j],
93*b1cdbd2cSJim Jagielski             const_cast< void * >(
94*b1cdbd2cSJim Jagielski                 i++->getValue(css::uno::TypeDescription(ctd->ppTypeRefs[j]))),
95*b1cdbd2cSJim Jagielski             ctd->ppTypeRefs[j], 0);
96*b1cdbd2cSJim Jagielski     }
97*b1cdbd2cSJim Jagielski     return i;
98*b1cdbd2cSJim Jagielski }
99*b1cdbd2cSJim Jagielski 
100*b1cdbd2cSJim Jagielski }
101*b1cdbd2cSJim Jagielski 
Unmarshal(rtl::Reference<Bridge> const & bridge,ReaderState & state,css::uno::Sequence<sal_Int8> const & buffer)102*b1cdbd2cSJim Jagielski Unmarshal::Unmarshal(
103*b1cdbd2cSJim Jagielski     rtl::Reference< Bridge > const & bridge, ReaderState & state,
104*b1cdbd2cSJim Jagielski     css::uno::Sequence< sal_Int8 > const & buffer):
105*b1cdbd2cSJim Jagielski     bridge_(bridge), state_(state), buffer_(buffer)
106*b1cdbd2cSJim Jagielski {
107*b1cdbd2cSJim Jagielski     data_ = reinterpret_cast< sal_uInt8 const * >(buffer_.getConstArray());
108*b1cdbd2cSJim Jagielski     end_ = data_ + buffer_.getLength();
109*b1cdbd2cSJim Jagielski }
110*b1cdbd2cSJim Jagielski 
~Unmarshal()111*b1cdbd2cSJim Jagielski Unmarshal::~Unmarshal() {}
112*b1cdbd2cSJim Jagielski 
read8()113*b1cdbd2cSJim Jagielski sal_uInt8 Unmarshal::read8() {
114*b1cdbd2cSJim Jagielski     check(1);
115*b1cdbd2cSJim Jagielski     return *data_++;
116*b1cdbd2cSJim Jagielski }
117*b1cdbd2cSJim Jagielski 
read16()118*b1cdbd2cSJim Jagielski sal_uInt16 Unmarshal::read16() {
119*b1cdbd2cSJim Jagielski     check(2);
120*b1cdbd2cSJim Jagielski     sal_uInt16 n = static_cast< sal_uInt16 >(*data_++) << 8;
121*b1cdbd2cSJim Jagielski     return n | *data_++;
122*b1cdbd2cSJim Jagielski }
123*b1cdbd2cSJim Jagielski 
read32()124*b1cdbd2cSJim Jagielski sal_uInt32 Unmarshal::read32() {
125*b1cdbd2cSJim Jagielski     check(4);
126*b1cdbd2cSJim Jagielski     sal_uInt32 n = static_cast< sal_uInt32 >(*data_++) << 24;
127*b1cdbd2cSJim Jagielski     n |= static_cast< sal_uInt32 >(*data_++) << 16;
128*b1cdbd2cSJim Jagielski     n |= static_cast< sal_uInt32 >(*data_++) << 8;
129*b1cdbd2cSJim Jagielski     return n | *data_++;
130*b1cdbd2cSJim Jagielski }
131*b1cdbd2cSJim Jagielski 
readType()132*b1cdbd2cSJim Jagielski css::uno::TypeDescription Unmarshal::readType() {
133*b1cdbd2cSJim Jagielski     sal_uInt8 flags = read8();
134*b1cdbd2cSJim Jagielski     typelib_TypeClass tc = static_cast< typelib_TypeClass >(flags & 0x7F);
135*b1cdbd2cSJim Jagielski     switch (tc) {
136*b1cdbd2cSJim Jagielski     case typelib_TypeClass_VOID:
137*b1cdbd2cSJim Jagielski     case typelib_TypeClass_BOOLEAN:
138*b1cdbd2cSJim Jagielski     case typelib_TypeClass_BYTE:
139*b1cdbd2cSJim Jagielski     case typelib_TypeClass_SHORT:
140*b1cdbd2cSJim Jagielski     case typelib_TypeClass_UNSIGNED_SHORT:
141*b1cdbd2cSJim Jagielski     case typelib_TypeClass_LONG:
142*b1cdbd2cSJim Jagielski     case typelib_TypeClass_UNSIGNED_LONG:
143*b1cdbd2cSJim Jagielski     case typelib_TypeClass_HYPER:
144*b1cdbd2cSJim Jagielski     case typelib_TypeClass_UNSIGNED_HYPER:
145*b1cdbd2cSJim Jagielski     case typelib_TypeClass_FLOAT:
146*b1cdbd2cSJim Jagielski     case typelib_TypeClass_DOUBLE:
147*b1cdbd2cSJim Jagielski     case typelib_TypeClass_CHAR:
148*b1cdbd2cSJim Jagielski     case typelib_TypeClass_STRING:
149*b1cdbd2cSJim Jagielski     case typelib_TypeClass_TYPE:
150*b1cdbd2cSJim Jagielski     case typelib_TypeClass_ANY:
151*b1cdbd2cSJim Jagielski         if ((flags & 0x80) != 0) {
152*b1cdbd2cSJim Jagielski             throw css::io::IOException(
153*b1cdbd2cSJim Jagielski                 rtl::OUString(
154*b1cdbd2cSJim Jagielski                     RTL_CONSTASCII_USTRINGPARAM(
155*b1cdbd2cSJim Jagielski                         "binaryurp::Unmarshal: cache flag of simple type is"
156*b1cdbd2cSJim Jagielski                         " set")),
157*b1cdbd2cSJim Jagielski                 css::uno::Reference< css::uno::XInterface >());
158*b1cdbd2cSJim Jagielski         }
159*b1cdbd2cSJim Jagielski         return css::uno::TypeDescription(
160*b1cdbd2cSJim Jagielski             *typelib_static_type_getByTypeClass(
161*b1cdbd2cSJim Jagielski                 static_cast< typelib_TypeClass >(tc)));
162*b1cdbd2cSJim Jagielski     case typelib_TypeClass_SEQUENCE:
163*b1cdbd2cSJim Jagielski     case typelib_TypeClass_ENUM:
164*b1cdbd2cSJim Jagielski     case typelib_TypeClass_STRUCT:
165*b1cdbd2cSJim Jagielski     case typelib_TypeClass_EXCEPTION:
166*b1cdbd2cSJim Jagielski     case typelib_TypeClass_INTERFACE:
167*b1cdbd2cSJim Jagielski         {
168*b1cdbd2cSJim Jagielski             sal_uInt16 idx = readCacheIndex();
169*b1cdbd2cSJim Jagielski             if ((flags & 0x80) == 0) {
170*b1cdbd2cSJim Jagielski                 if (idx == cache::ignore || !state_.typeCache[idx].is()) {
171*b1cdbd2cSJim Jagielski                     throw css::io::IOException(
172*b1cdbd2cSJim Jagielski                         rtl::OUString(
173*b1cdbd2cSJim Jagielski                             RTL_CONSTASCII_USTRINGPARAM(
174*b1cdbd2cSJim Jagielski                                 "binaryurp::Unmarshal: unknown type cache"
175*b1cdbd2cSJim Jagielski                                 " index")),
176*b1cdbd2cSJim Jagielski                         css::uno::Reference< css::uno::XInterface >());
177*b1cdbd2cSJim Jagielski                 }
178*b1cdbd2cSJim Jagielski                 return state_.typeCache[idx];
179*b1cdbd2cSJim Jagielski             } else {
180*b1cdbd2cSJim Jagielski                 css::uno::TypeDescription t(readString());
181*b1cdbd2cSJim Jagielski                 if (!t.is() ||
182*b1cdbd2cSJim Jagielski                     t.get()->eTypeClass != static_cast< typelib_TypeClass >(tc))
183*b1cdbd2cSJim Jagielski                 {
184*b1cdbd2cSJim Jagielski                     throw css::io::IOException(
185*b1cdbd2cSJim Jagielski                         rtl::OUString(
186*b1cdbd2cSJim Jagielski                             RTL_CONSTASCII_USTRINGPARAM(
187*b1cdbd2cSJim Jagielski                                 "binaryurp::Unmarshal: type with unknown"
188*b1cdbd2cSJim Jagielski                                 " name")),
189*b1cdbd2cSJim Jagielski                         css::uno::Reference< css::uno::XInterface >());
190*b1cdbd2cSJim Jagielski                 }
191*b1cdbd2cSJim Jagielski                 for (css::uno::TypeDescription t2(t);
192*b1cdbd2cSJim Jagielski                      t2.get()->eTypeClass == typelib_TypeClass_SEQUENCE;)
193*b1cdbd2cSJim Jagielski                 {
194*b1cdbd2cSJim Jagielski                     t2.makeComplete();
195*b1cdbd2cSJim Jagielski                     t2 = css::uno::TypeDescription(
196*b1cdbd2cSJim Jagielski                         reinterpret_cast< typelib_IndirectTypeDescription * >(
197*b1cdbd2cSJim Jagielski                             t2.get())->pType);
198*b1cdbd2cSJim Jagielski                     if (!t2.is()) {
199*b1cdbd2cSJim Jagielski                         throw css::io::IOException(
200*b1cdbd2cSJim Jagielski                             rtl::OUString(
201*b1cdbd2cSJim Jagielski                                 RTL_CONSTASCII_USTRINGPARAM(
202*b1cdbd2cSJim Jagielski                                     "binaryurp::Unmarshal: sequence type with"
203*b1cdbd2cSJim Jagielski                                     " unknown component type")),
204*b1cdbd2cSJim Jagielski                             css::uno::Reference< css::uno::XInterface >());
205*b1cdbd2cSJim Jagielski                     }
206*b1cdbd2cSJim Jagielski                     switch (t2.get()->eTypeClass) {
207*b1cdbd2cSJim Jagielski                     case typelib_TypeClass_VOID:
208*b1cdbd2cSJim Jagielski                     case typelib_TypeClass_EXCEPTION:
209*b1cdbd2cSJim Jagielski                         throw css::io::IOException(
210*b1cdbd2cSJim Jagielski                             rtl::OUString(
211*b1cdbd2cSJim Jagielski                                 RTL_CONSTASCII_USTRINGPARAM(
212*b1cdbd2cSJim Jagielski                                     "binaryurp::Unmarshal: sequence type with"
213*b1cdbd2cSJim Jagielski                                     " bad component type")),
214*b1cdbd2cSJim Jagielski                             css::uno::Reference< css::uno::XInterface >());
215*b1cdbd2cSJim Jagielski                     default:
216*b1cdbd2cSJim Jagielski                         break;
217*b1cdbd2cSJim Jagielski                     }
218*b1cdbd2cSJim Jagielski                 }
219*b1cdbd2cSJim Jagielski                 if (idx != cache::ignore) {
220*b1cdbd2cSJim Jagielski                     state_.typeCache[idx] = t;
221*b1cdbd2cSJim Jagielski                 }
222*b1cdbd2cSJim Jagielski                 return t;
223*b1cdbd2cSJim Jagielski             }
224*b1cdbd2cSJim Jagielski         }
225*b1cdbd2cSJim Jagielski     default:
226*b1cdbd2cSJim Jagielski         throw css::io::IOException(
227*b1cdbd2cSJim Jagielski             rtl::OUString(
228*b1cdbd2cSJim Jagielski                 RTL_CONSTASCII_USTRINGPARAM(
229*b1cdbd2cSJim Jagielski                     "binaryurp::Unmarshal: type of unknown type class")),
230*b1cdbd2cSJim Jagielski             css::uno::Reference< css::uno::XInterface >());
231*b1cdbd2cSJim Jagielski     }
232*b1cdbd2cSJim Jagielski }
233*b1cdbd2cSJim Jagielski 
readOid()234*b1cdbd2cSJim Jagielski rtl::OUString Unmarshal::readOid() {
235*b1cdbd2cSJim Jagielski     rtl::OUString oid(readString());
236*b1cdbd2cSJim Jagielski     for (sal_Int32 i = 0; i != oid.getLength(); ++i) {
237*b1cdbd2cSJim Jagielski         if (oid[i] > 0x7F) {
238*b1cdbd2cSJim Jagielski             throw css::io::IOException(
239*b1cdbd2cSJim Jagielski                 rtl::OUString(
240*b1cdbd2cSJim Jagielski                     RTL_CONSTASCII_USTRINGPARAM(
241*b1cdbd2cSJim Jagielski                         "binaryurp::Unmarshal: OID contains non-ASCII"
242*b1cdbd2cSJim Jagielski                         " character")),
243*b1cdbd2cSJim Jagielski                 css::uno::Reference< css::uno::XInterface >());
244*b1cdbd2cSJim Jagielski         }
245*b1cdbd2cSJim Jagielski     }
246*b1cdbd2cSJim Jagielski     sal_uInt16 idx = readCacheIndex();
247*b1cdbd2cSJim Jagielski     if (oid.isEmpty() && idx != cache::ignore) {
248*b1cdbd2cSJim Jagielski         if ( state_.oidCache[idx].isEmpty() ) {
249*b1cdbd2cSJim Jagielski             throw css::io::IOException(
250*b1cdbd2cSJim Jagielski                 rtl::OUString(
251*b1cdbd2cSJim Jagielski                     RTL_CONSTASCII_USTRINGPARAM(
252*b1cdbd2cSJim Jagielski                         "binaryurp::Unmarshal: unknown OID cache index")),
253*b1cdbd2cSJim Jagielski                 css::uno::Reference< css::uno::XInterface >());
254*b1cdbd2cSJim Jagielski         }
255*b1cdbd2cSJim Jagielski         return state_.oidCache[idx];
256*b1cdbd2cSJim Jagielski     }
257*b1cdbd2cSJim Jagielski     if (idx != cache::ignore) {
258*b1cdbd2cSJim Jagielski         state_.oidCache[idx] = oid;
259*b1cdbd2cSJim Jagielski     }
260*b1cdbd2cSJim Jagielski     return oid;
261*b1cdbd2cSJim Jagielski }
262*b1cdbd2cSJim Jagielski 
readTid()263*b1cdbd2cSJim Jagielski rtl::ByteSequence Unmarshal::readTid() {
264*b1cdbd2cSJim Jagielski     rtl::ByteSequence tid(
265*b1cdbd2cSJim Jagielski         *static_cast< sal_Sequence * const * >(
266*b1cdbd2cSJim Jagielski             readSequence(
267*b1cdbd2cSJim Jagielski                 css::uno::TypeDescription(
268*b1cdbd2cSJim Jagielski                     cppu::UnoType< css::uno::Sequence< sal_Int8 > >::get())).
269*b1cdbd2cSJim Jagielski             getValue(
270*b1cdbd2cSJim Jagielski                 css::uno::TypeDescription(
271*b1cdbd2cSJim Jagielski                     cppu::UnoType< css::uno::Sequence< sal_Int8 > >::get()))));
272*b1cdbd2cSJim Jagielski     sal_uInt16 idx = readCacheIndex();
273*b1cdbd2cSJim Jagielski     if (tid.getLength() == 0) {
274*b1cdbd2cSJim Jagielski         if (idx == cache::ignore || state_.tidCache[idx].getLength() == 0) {
275*b1cdbd2cSJim Jagielski             throw css::io::IOException(
276*b1cdbd2cSJim Jagielski                 rtl::OUString(
277*b1cdbd2cSJim Jagielski                     RTL_CONSTASCII_USTRINGPARAM(
278*b1cdbd2cSJim Jagielski                         "binaryurp::Unmarshal: unknown TID cache index")),
279*b1cdbd2cSJim Jagielski                 css::uno::Reference< css::uno::XInterface >());
280*b1cdbd2cSJim Jagielski         }
281*b1cdbd2cSJim Jagielski         return state_.tidCache[idx];
282*b1cdbd2cSJim Jagielski     }
283*b1cdbd2cSJim Jagielski     if (idx != cache::ignore) {
284*b1cdbd2cSJim Jagielski         state_.tidCache[idx] = tid;
285*b1cdbd2cSJim Jagielski     }
286*b1cdbd2cSJim Jagielski     return tid;
287*b1cdbd2cSJim Jagielski }
288*b1cdbd2cSJim Jagielski 
readValue(css::uno::TypeDescription const & type)289*b1cdbd2cSJim Jagielski BinaryAny Unmarshal::readValue(css::uno::TypeDescription const & type) {
290*b1cdbd2cSJim Jagielski     OSL_ASSERT(type.is());
291*b1cdbd2cSJim Jagielski     switch (type.get()->eTypeClass) {
292*b1cdbd2cSJim Jagielski     default:
293*b1cdbd2cSJim Jagielski         std::abort(); // this cannot happen
294*b1cdbd2cSJim Jagielski         // pseudo fall-through to avoid compiler warnings
295*b1cdbd2cSJim Jagielski     case typelib_TypeClass_VOID:
296*b1cdbd2cSJim Jagielski         return BinaryAny();
297*b1cdbd2cSJim Jagielski     case typelib_TypeClass_BOOLEAN:
298*b1cdbd2cSJim Jagielski         {
299*b1cdbd2cSJim Jagielski             sal_uInt8 v = read8();
300*b1cdbd2cSJim Jagielski             if (v > 1) {
301*b1cdbd2cSJim Jagielski                 throw css::io::IOException(
302*b1cdbd2cSJim Jagielski                     rtl::OUString(
303*b1cdbd2cSJim Jagielski                         RTL_CONSTASCII_USTRINGPARAM(
304*b1cdbd2cSJim Jagielski                             "binaryurp::Unmarshal: boolean of unknown value")),
305*b1cdbd2cSJim Jagielski                     css::uno::Reference< css::uno::XInterface >());
306*b1cdbd2cSJim Jagielski             }
307*b1cdbd2cSJim Jagielski             return BinaryAny(type, &v);
308*b1cdbd2cSJim Jagielski         }
309*b1cdbd2cSJim Jagielski     case typelib_TypeClass_BYTE:
310*b1cdbd2cSJim Jagielski         {
311*b1cdbd2cSJim Jagielski             sal_uInt8 v = read8();
312*b1cdbd2cSJim Jagielski             return BinaryAny(type, &v);
313*b1cdbd2cSJim Jagielski         }
314*b1cdbd2cSJim Jagielski     case typelib_TypeClass_SHORT:
315*b1cdbd2cSJim Jagielski     case typelib_TypeClass_UNSIGNED_SHORT:
316*b1cdbd2cSJim Jagielski     case typelib_TypeClass_CHAR:
317*b1cdbd2cSJim Jagielski         {
318*b1cdbd2cSJim Jagielski             sal_uInt16 v = read16();
319*b1cdbd2cSJim Jagielski             return BinaryAny(type, &v);
320*b1cdbd2cSJim Jagielski         }
321*b1cdbd2cSJim Jagielski     case typelib_TypeClass_LONG:
322*b1cdbd2cSJim Jagielski     case typelib_TypeClass_UNSIGNED_LONG:
323*b1cdbd2cSJim Jagielski     case typelib_TypeClass_FLOAT:
324*b1cdbd2cSJim Jagielski         {
325*b1cdbd2cSJim Jagielski             sal_uInt32 v = read32();
326*b1cdbd2cSJim Jagielski             return BinaryAny(type, &v);
327*b1cdbd2cSJim Jagielski         }
328*b1cdbd2cSJim Jagielski     case typelib_TypeClass_HYPER:
329*b1cdbd2cSJim Jagielski     case typelib_TypeClass_UNSIGNED_HYPER:
330*b1cdbd2cSJim Jagielski     case typelib_TypeClass_DOUBLE:
331*b1cdbd2cSJim Jagielski         {
332*b1cdbd2cSJim Jagielski             sal_uInt64 v = read64();
333*b1cdbd2cSJim Jagielski             return BinaryAny(type, &v);
334*b1cdbd2cSJim Jagielski         }
335*b1cdbd2cSJim Jagielski     case typelib_TypeClass_STRING:
336*b1cdbd2cSJim Jagielski         {
337*b1cdbd2cSJim Jagielski             rtl::OUString v(readString());
338*b1cdbd2cSJim Jagielski             return BinaryAny(type, &v.pData);
339*b1cdbd2cSJim Jagielski         }
340*b1cdbd2cSJim Jagielski     case typelib_TypeClass_TYPE:
341*b1cdbd2cSJim Jagielski         {
342*b1cdbd2cSJim Jagielski             css::uno::TypeDescription v(readType());
343*b1cdbd2cSJim Jagielski             typelib_TypeDescription * p = v.get();
344*b1cdbd2cSJim Jagielski             return BinaryAny(type, &p);
345*b1cdbd2cSJim Jagielski         }
346*b1cdbd2cSJim Jagielski     case typelib_TypeClass_ANY:
347*b1cdbd2cSJim Jagielski         {
348*b1cdbd2cSJim Jagielski             css::uno::TypeDescription t(readType());
349*b1cdbd2cSJim Jagielski             if (t.get()->eTypeClass == typelib_TypeClass_ANY) {
350*b1cdbd2cSJim Jagielski                 throw css::io::IOException(
351*b1cdbd2cSJim Jagielski                     rtl::OUString(
352*b1cdbd2cSJim Jagielski                         RTL_CONSTASCII_USTRINGPARAM(
353*b1cdbd2cSJim Jagielski                             "binaryurp::Unmarshal: any of type ANY")),
354*b1cdbd2cSJim Jagielski                     css::uno::Reference< css::uno::XInterface >());
355*b1cdbd2cSJim Jagielski             }
356*b1cdbd2cSJim Jagielski             return readValue(t);
357*b1cdbd2cSJim Jagielski         }
358*b1cdbd2cSJim Jagielski     case typelib_TypeClass_SEQUENCE:
359*b1cdbd2cSJim Jagielski         type.makeComplete();
360*b1cdbd2cSJim Jagielski         return readSequence(type);
361*b1cdbd2cSJim Jagielski     case typelib_TypeClass_ENUM:
362*b1cdbd2cSJim Jagielski         {
363*b1cdbd2cSJim Jagielski             sal_Int32 v = static_cast< sal_Int32 >(read32());
364*b1cdbd2cSJim Jagielski             type.makeComplete();
365*b1cdbd2cSJim Jagielski             typelib_EnumTypeDescription * etd =
366*b1cdbd2cSJim Jagielski                 reinterpret_cast< typelib_EnumTypeDescription * >(type.get());
367*b1cdbd2cSJim Jagielski             bool found = false;
368*b1cdbd2cSJim Jagielski             for (sal_Int32 i = 0; i != etd->nEnumValues; ++i) {
369*b1cdbd2cSJim Jagielski                 if (etd->pEnumValues[i] == v) {
370*b1cdbd2cSJim Jagielski                     found = true;
371*b1cdbd2cSJim Jagielski                     break;
372*b1cdbd2cSJim Jagielski                 }
373*b1cdbd2cSJim Jagielski             }
374*b1cdbd2cSJim Jagielski             if (!found) {
375*b1cdbd2cSJim Jagielski                 throw css::io::IOException(
376*b1cdbd2cSJim Jagielski                     rtl::OUString(
377*b1cdbd2cSJim Jagielski                         RTL_CONSTASCII_USTRINGPARAM(
378*b1cdbd2cSJim Jagielski                             "binaryurp::Unmarshal: unknown enum value")),
379*b1cdbd2cSJim Jagielski                     css::uno::Reference< css::uno::XInterface >());
380*b1cdbd2cSJim Jagielski             }
381*b1cdbd2cSJim Jagielski             return BinaryAny(type, &v);
382*b1cdbd2cSJim Jagielski         }
383*b1cdbd2cSJim Jagielski     case typelib_TypeClass_STRUCT:
384*b1cdbd2cSJim Jagielski     case typelib_TypeClass_EXCEPTION:
385*b1cdbd2cSJim Jagielski         {
386*b1cdbd2cSJim Jagielski             std::vector< BinaryAny > as;
387*b1cdbd2cSJim Jagielski             readMemberValues(type, &as);
388*b1cdbd2cSJim Jagielski             void * buf = allocate(type.get()->nSize);
389*b1cdbd2cSJim Jagielski             copyMemberValues(type, as.begin(), buf);
390*b1cdbd2cSJim Jagielski             uno_Any raw;
391*b1cdbd2cSJim Jagielski             raw.pType = reinterpret_cast< typelib_TypeDescriptionReference * >(
392*b1cdbd2cSJim Jagielski                 type.get());
393*b1cdbd2cSJim Jagielski             raw.pData = buf;
394*b1cdbd2cSJim Jagielski             raw.pReserved = 0;
395*b1cdbd2cSJim Jagielski             return BinaryAny(raw);
396*b1cdbd2cSJim Jagielski         }
397*b1cdbd2cSJim Jagielski     case typelib_TypeClass_INTERFACE:
398*b1cdbd2cSJim Jagielski         {
399*b1cdbd2cSJim Jagielski             css::uno::UnoInterfaceReference obj(
400*b1cdbd2cSJim Jagielski                 bridge_->registerIncomingInterface(readOid(), type));
401*b1cdbd2cSJim Jagielski             return BinaryAny(type, &obj.m_pUnoI);
402*b1cdbd2cSJim Jagielski         }
403*b1cdbd2cSJim Jagielski     }
404*b1cdbd2cSJim Jagielski }
405*b1cdbd2cSJim Jagielski 
done() const406*b1cdbd2cSJim Jagielski void Unmarshal::done() const {
407*b1cdbd2cSJim Jagielski     if (data_ != end_) {
408*b1cdbd2cSJim Jagielski         throw css::io::IOException(
409*b1cdbd2cSJim Jagielski             rtl::OUString(
410*b1cdbd2cSJim Jagielski                 RTL_CONSTASCII_USTRINGPARAM(
411*b1cdbd2cSJim Jagielski                     "binaryurp::Unmarshal: block contains excess data")),
412*b1cdbd2cSJim Jagielski             css::uno::Reference< css::uno::XInterface >());
413*b1cdbd2cSJim Jagielski     }
414*b1cdbd2cSJim Jagielski }
415*b1cdbd2cSJim Jagielski 
check(sal_Int32 size) const416*b1cdbd2cSJim Jagielski void Unmarshal::check(sal_Int32 size) const {
417*b1cdbd2cSJim Jagielski     if (end_ - data_ < size) {
418*b1cdbd2cSJim Jagielski         throw css::io::IOException(
419*b1cdbd2cSJim Jagielski             rtl::OUString(
420*b1cdbd2cSJim Jagielski                 RTL_CONSTASCII_USTRINGPARAM(
421*b1cdbd2cSJim Jagielski                     "binaryurp::Unmarshal: trying to read past end of block")),
422*b1cdbd2cSJim Jagielski             css::uno::Reference< css::uno::XInterface >());
423*b1cdbd2cSJim Jagielski     }
424*b1cdbd2cSJim Jagielski }
425*b1cdbd2cSJim Jagielski 
readCompressed()426*b1cdbd2cSJim Jagielski sal_uInt32 Unmarshal::readCompressed() {
427*b1cdbd2cSJim Jagielski     sal_uInt8 n = read8();
428*b1cdbd2cSJim Jagielski     return n == 0xFF ? read32() : n;
429*b1cdbd2cSJim Jagielski }
430*b1cdbd2cSJim Jagielski 
readCacheIndex()431*b1cdbd2cSJim Jagielski sal_uInt16 Unmarshal::readCacheIndex() {
432*b1cdbd2cSJim Jagielski     sal_uInt16 idx = read16();
433*b1cdbd2cSJim Jagielski     if (idx >= cache::size && idx != cache::ignore) {
434*b1cdbd2cSJim Jagielski         throw css::io::IOException(
435*b1cdbd2cSJim Jagielski             rtl::OUString(
436*b1cdbd2cSJim Jagielski                 RTL_CONSTASCII_USTRINGPARAM(
437*b1cdbd2cSJim Jagielski                     "binaryurp::Unmarshal: cache index out of range")),
438*b1cdbd2cSJim Jagielski             css::uno::Reference< css::uno::XInterface >());
439*b1cdbd2cSJim Jagielski     }
440*b1cdbd2cSJim Jagielski     return idx;
441*b1cdbd2cSJim Jagielski }
442*b1cdbd2cSJim Jagielski 
read64()443*b1cdbd2cSJim Jagielski sal_uInt64 Unmarshal::read64() {
444*b1cdbd2cSJim Jagielski     check(8);
445*b1cdbd2cSJim Jagielski     sal_uInt64 n = static_cast< sal_uInt64 >(*data_++) << 56;
446*b1cdbd2cSJim Jagielski     n |= static_cast< sal_uInt64 >(*data_++) << 48;
447*b1cdbd2cSJim Jagielski     n |= static_cast< sal_uInt64 >(*data_++) << 40;
448*b1cdbd2cSJim Jagielski     n |= static_cast< sal_uInt64 >(*data_++) << 32;
449*b1cdbd2cSJim Jagielski     n |= static_cast< sal_uInt64 >(*data_++) << 24;
450*b1cdbd2cSJim Jagielski     n |= static_cast< sal_uInt64 >(*data_++) << 16;
451*b1cdbd2cSJim Jagielski     n |= static_cast< sal_uInt64 >(*data_++) << 8;
452*b1cdbd2cSJim Jagielski     return n | *data_++;
453*b1cdbd2cSJim Jagielski }
454*b1cdbd2cSJim Jagielski 
readString()455*b1cdbd2cSJim Jagielski rtl::OUString Unmarshal::readString() {
456*b1cdbd2cSJim Jagielski     sal_uInt32 n = readCompressed();
457*b1cdbd2cSJim Jagielski     if (n > SAL_MAX_INT32) {
458*b1cdbd2cSJim Jagielski         throw css::uno::RuntimeException(
459*b1cdbd2cSJim Jagielski             rtl::OUString(
460*b1cdbd2cSJim Jagielski                 RTL_CONSTASCII_USTRINGPARAM(
461*b1cdbd2cSJim Jagielski                     "binaryurp::Unmarshal: string size too large")),
462*b1cdbd2cSJim Jagielski             css::uno::Reference< css::uno::XInterface >());
463*b1cdbd2cSJim Jagielski     }
464*b1cdbd2cSJim Jagielski     check(static_cast< sal_Int32 >(n));
465*b1cdbd2cSJim Jagielski     rtl::OUString s;
466*b1cdbd2cSJim Jagielski     if (!rtl_convertStringToUString(
467*b1cdbd2cSJim Jagielski             &s.pData, reinterpret_cast< char const * >(data_),
468*b1cdbd2cSJim Jagielski             static_cast< sal_Int32 >(n), RTL_TEXTENCODING_UTF8,
469*b1cdbd2cSJim Jagielski             (RTL_TEXTTOUNICODE_FLAGS_UNDEFINED_ERROR |
470*b1cdbd2cSJim Jagielski              RTL_TEXTTOUNICODE_FLAGS_MBUNDEFINED_ERROR |
471*b1cdbd2cSJim Jagielski              RTL_TEXTTOUNICODE_FLAGS_INVALID_ERROR)))
472*b1cdbd2cSJim Jagielski     {
473*b1cdbd2cSJim Jagielski         throw css::io::IOException(
474*b1cdbd2cSJim Jagielski             rtl::OUString(
475*b1cdbd2cSJim Jagielski                 RTL_CONSTASCII_USTRINGPARAM(
476*b1cdbd2cSJim Jagielski                     "binaryurp::Unmarshal: string does not contain UTF-8")),
477*b1cdbd2cSJim Jagielski             css::uno::Reference< css::uno::XInterface >());
478*b1cdbd2cSJim Jagielski     }
479*b1cdbd2cSJim Jagielski     data_ += n;
480*b1cdbd2cSJim Jagielski     return s;
481*b1cdbd2cSJim Jagielski }
482*b1cdbd2cSJim Jagielski 
readSequence(css::uno::TypeDescription const & type)483*b1cdbd2cSJim Jagielski BinaryAny Unmarshal::readSequence(css::uno::TypeDescription const & type) {
484*b1cdbd2cSJim Jagielski     OSL_ASSERT(
485*b1cdbd2cSJim Jagielski         type.is() && type.get()->eTypeClass == typelib_TypeClass_SEQUENCE);
486*b1cdbd2cSJim Jagielski     sal_uInt32 n = readCompressed();
487*b1cdbd2cSJim Jagielski     if (n > SAL_MAX_INT32) {
488*b1cdbd2cSJim Jagielski         throw css::uno::RuntimeException(
489*b1cdbd2cSJim Jagielski             rtl::OUString(
490*b1cdbd2cSJim Jagielski                 RTL_CONSTASCII_USTRINGPARAM(
491*b1cdbd2cSJim Jagielski                     "binaryurp::Unmarshal: sequence size too large")),
492*b1cdbd2cSJim Jagielski             css::uno::Reference< css::uno::XInterface >());
493*b1cdbd2cSJim Jagielski     }
494*b1cdbd2cSJim Jagielski     if (n == 0) {
495*b1cdbd2cSJim Jagielski         return BinaryAny(type, 0);
496*b1cdbd2cSJim Jagielski     }
497*b1cdbd2cSJim Jagielski     css::uno::TypeDescription ctd(
498*b1cdbd2cSJim Jagielski         reinterpret_cast< typelib_IndirectTypeDescription * >(
499*b1cdbd2cSJim Jagielski             type.get())->pType);
500*b1cdbd2cSJim Jagielski     if (ctd.get()->eTypeClass == typelib_TypeClass_BYTE) {
501*b1cdbd2cSJim Jagielski         check(static_cast< sal_Int32 >(n));
502*b1cdbd2cSJim Jagielski         rtl::ByteSequence s(
503*b1cdbd2cSJim Jagielski             reinterpret_cast< sal_Int8 const * >(data_),
504*b1cdbd2cSJim Jagielski             static_cast< sal_Int32 >(n));
505*b1cdbd2cSJim Jagielski         data_ += n;
506*b1cdbd2cSJim Jagielski         sal_Sequence * p = s.getHandle();
507*b1cdbd2cSJim Jagielski         return BinaryAny(type, &p);
508*b1cdbd2cSJim Jagielski     }
509*b1cdbd2cSJim Jagielski     std::vector< BinaryAny > as;
510*b1cdbd2cSJim Jagielski     for (sal_uInt32 i = 0; i != n; ++i) {
511*b1cdbd2cSJim Jagielski         as.push_back(readValue(ctd));
512*b1cdbd2cSJim Jagielski     }
513*b1cdbd2cSJim Jagielski     OSL_ASSERT(ctd.get()->nSize >= 0);
514*b1cdbd2cSJim Jagielski     sal_uInt64 size = static_cast< sal_uInt64 >(n) *
515*b1cdbd2cSJim Jagielski         static_cast< sal_uInt64 >(ctd.get()->nSize);
516*b1cdbd2cSJim Jagielski         // sal_uInt32 * sal_Int32 -> sal_uInt64 cannot overflow
517*b1cdbd2cSJim Jagielski     if (size > SAL_MAX_SIZE - SAL_SEQUENCE_HEADER_SIZE) {
518*b1cdbd2cSJim Jagielski         throw css::uno::RuntimeException(
519*b1cdbd2cSJim Jagielski             rtl::OUString(
520*b1cdbd2cSJim Jagielski                 RTL_CONSTASCII_USTRINGPARAM(
521*b1cdbd2cSJim Jagielski                     "binaryurp::Unmarshal: sequence size too large")),
522*b1cdbd2cSJim Jagielski             css::uno::Reference< css::uno::XInterface >());
523*b1cdbd2cSJim Jagielski     }
524*b1cdbd2cSJim Jagielski     void * buf = allocate(
525*b1cdbd2cSJim Jagielski         SAL_SEQUENCE_HEADER_SIZE + static_cast< sal_Size >(size));
526*b1cdbd2cSJim Jagielski     static_cast< sal_Sequence * >(buf)->nRefCount = 0;
527*b1cdbd2cSJim Jagielski     static_cast< sal_Sequence * >(buf)->nElements =
528*b1cdbd2cSJim Jagielski         static_cast< sal_Int32 >(n);
529*b1cdbd2cSJim Jagielski     for (sal_uInt32 i = 0; i != n; ++i) {
530*b1cdbd2cSJim Jagielski         uno_copyData(
531*b1cdbd2cSJim Jagielski             static_cast< sal_Sequence * >(buf)->elements + i * ctd.get()->nSize,
532*b1cdbd2cSJim Jagielski             const_cast< void * >(as[i].getValue(ctd)), ctd.get(), 0);
533*b1cdbd2cSJim Jagielski     }
534*b1cdbd2cSJim Jagielski     return BinaryAny(type, reinterpret_cast< sal_Sequence ** >(&buf));
535*b1cdbd2cSJim Jagielski }
536*b1cdbd2cSJim Jagielski 
readMemberValues(css::uno::TypeDescription const & type,std::vector<BinaryAny> * values)537*b1cdbd2cSJim Jagielski void Unmarshal::readMemberValues(
538*b1cdbd2cSJim Jagielski     css::uno::TypeDescription const & type, std::vector< BinaryAny > * values)
539*b1cdbd2cSJim Jagielski {
540*b1cdbd2cSJim Jagielski     OSL_ASSERT(
541*b1cdbd2cSJim Jagielski         type.is() &&
542*b1cdbd2cSJim Jagielski         (type.get()->eTypeClass == typelib_TypeClass_STRUCT ||
543*b1cdbd2cSJim Jagielski          type.get()->eTypeClass == typelib_TypeClass_EXCEPTION) &&
544*b1cdbd2cSJim Jagielski         values != 0);
545*b1cdbd2cSJim Jagielski     type.makeComplete();
546*b1cdbd2cSJim Jagielski     typelib_CompoundTypeDescription * ctd =
547*b1cdbd2cSJim Jagielski         reinterpret_cast< typelib_CompoundTypeDescription * >(type.get());
548*b1cdbd2cSJim Jagielski     if (ctd->pBaseTypeDescription != 0) {
549*b1cdbd2cSJim Jagielski         readMemberValues(
550*b1cdbd2cSJim Jagielski             css::uno::TypeDescription(&ctd->pBaseTypeDescription->aBase),
551*b1cdbd2cSJim Jagielski             values);
552*b1cdbd2cSJim Jagielski     }
553*b1cdbd2cSJim Jagielski     for (sal_Int32 i = 0; i != ctd->nMembers; ++i) {
554*b1cdbd2cSJim Jagielski         values->push_back(
555*b1cdbd2cSJim Jagielski             readValue(css::uno::TypeDescription(ctd->ppTypeRefs[i])));
556*b1cdbd2cSJim Jagielski     }
557*b1cdbd2cSJim Jagielski }
558*b1cdbd2cSJim Jagielski 
559*b1cdbd2cSJim Jagielski }
560