Line data Source code
1 : /* -*- Mode: C++; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4 -*- */
2 : /*
3 : * This file is part of the LibreOffice project.
4 : *
5 : * This Source Code Form is subject to the terms of the Mozilla Public
6 : * License, v. 2.0. If a copy of the MPL was not distributed with this
7 : * file, You can obtain one at http://mozilla.org/MPL/2.0/.
8 : *
9 : * This file incorporates work covered by the following license notice:
10 : *
11 : * Licensed to the Apache Software Foundation (ASF) under one or more
12 : * contributor license agreements. See the NOTICE file distributed
13 : * with this work for additional information regarding copyright
14 : * ownership. The ASF licenses this file to you under the Apache
15 : * License, Version 2.0 (the "License"); you may not use this file
16 : * except in compliance with the License. You may obtain a copy of
17 : * the License at http://www.apache.org/licenses/LICENSE-2.0 .
18 : */
19 :
20 : #include "sal/config.h"
21 :
22 : #include <exception>
23 : #include <vector>
24 : #include <string.h>
25 :
26 : #include "com/sun/star/connection/XConnection.hpp"
27 : #include "com/sun/star/lang/WrappedTargetRuntimeException.hpp"
28 : #include "com/sun/star/uno/XCurrentContext.hpp"
29 : #include "cppuhelper/exc_hlp.hxx"
30 : #include "osl/mutex.hxx"
31 : #include "uno/dispatcher.hxx"
32 :
33 : #include "binaryany.hxx"
34 : #include "bridge.hxx"
35 : #include "currentcontext.hxx"
36 : #include "specialfunctionids.hxx"
37 : #include "writer.hxx"
38 :
39 : namespace binaryurp {
40 :
41 0 : Writer::Item::Item() {}
42 :
43 0 : Writer::Item::Item(
44 : rtl::ByteSequence const & theTid, OUString const & theOid,
45 : css::uno::TypeDescription const & theType,
46 : css::uno::TypeDescription const & theMember,
47 : std::vector< BinaryAny > const & inArguments,
48 : css::uno::UnoInterfaceReference const & theCurrentContext):
49 : request(true), tid(theTid), oid(theOid), type(theType), member(theMember),
50 0 : arguments(inArguments), currentContext(theCurrentContext)
51 0 : {}
52 :
53 0 : Writer::Item::Item(
54 : rtl::ByteSequence const & theTid,
55 : css::uno::TypeDescription const & theMember, bool theSetter,
56 : bool theException, BinaryAny const & theReturnValue,
57 : std::vector< BinaryAny > const & outArguments,
58 : bool theSetCurrentContextMode):
59 : request(false), tid(theTid), member(theMember), setter(theSetter),
60 : arguments(outArguments), exception(theException),
61 0 : returnValue(theReturnValue), setCurrentContextMode(theSetCurrentContextMode)
62 0 : {}
63 :
64 0 : Writer::Writer(rtl::Reference< Bridge > const & bridge):
65 : Thread("binaryurpWriter"), bridge_(bridge), marshal_(bridge, state_),
66 0 : stop_(false)
67 : {
68 : OSL_ASSERT(bridge.is());
69 0 : }
70 :
71 0 : void Writer::sendDirectRequest(
72 : rtl::ByteSequence const & tid, OUString const & oid,
73 : css::uno::TypeDescription const & type,
74 : css::uno::TypeDescription const & member,
75 : std::vector< BinaryAny > const & inArguments)
76 : {
77 : OSL_ASSERT(!unblocked_.check());
78 : sendRequest(
79 : tid, oid, type, member, inArguments, false,
80 0 : css::uno::UnoInterfaceReference());
81 0 : }
82 :
83 0 : void Writer::sendDirectReply(
84 : rtl::ByteSequence const & tid, css::uno::TypeDescription const & member,
85 : bool exception, BinaryAny const & returnValue,
86 : std::vector< BinaryAny > const & outArguments)
87 : {
88 : OSL_ASSERT(!unblocked_.check());
89 0 : sendReply(tid, member, false, exception, returnValue,outArguments);
90 0 : }
91 :
92 0 : void Writer::queueRequest(
93 : rtl::ByteSequence const & tid, OUString const & oid,
94 : css::uno::TypeDescription const & type,
95 : css::uno::TypeDescription const & member,
96 : std::vector< BinaryAny > const & inArguments)
97 : {
98 0 : css::uno::UnoInterfaceReference cc(current_context::get());
99 0 : osl::MutexGuard g(mutex_);
100 0 : queue_.push_back(Item(tid, oid, type, member, inArguments, cc));
101 0 : items_.set();
102 0 : }
103 :
104 0 : void Writer::queueReply(
105 : rtl::ByteSequence const & tid,
106 : com::sun::star::uno::TypeDescription const & member, bool setter,
107 : bool exception, BinaryAny const & returnValue,
108 : std::vector< BinaryAny > const & outArguments, bool setCurrentContextMode)
109 : {
110 0 : osl::MutexGuard g(mutex_);
111 : queue_.push_back(
112 : Item(
113 : tid, member, setter, exception, returnValue, outArguments,
114 0 : setCurrentContextMode));
115 0 : items_.set();
116 0 : }
117 :
118 0 : void Writer::unblock() {
119 : // Assumes that osl::Condition::set works as a memory barrier, so that
120 : // changes made by preceeding sendDirectRequest/Reply calls are visible to
121 : // subsequent sendRequest/Reply calls:
122 0 : unblocked_.set();
123 0 : }
124 :
125 0 : void Writer::stop() {
126 : {
127 0 : osl::MutexGuard g(mutex_);
128 0 : stop_ = true;
129 : }
130 0 : unblocked_.set();
131 0 : items_.set();
132 0 : }
133 :
134 0 : Writer::~Writer() {}
135 :
136 0 : void Writer::execute() {
137 : try {
138 0 : unblocked_.wait();
139 0 : for (;;) {
140 0 : items_.wait();
141 0 : Item item;
142 : {
143 0 : osl::MutexGuard g(mutex_);
144 0 : if (stop_) {
145 0 : return;
146 : }
147 : OSL_ASSERT(!queue_.empty());
148 0 : item = queue_.front();
149 0 : queue_.pop_front();
150 0 : if (queue_.empty()) {
151 0 : items_.reset();
152 0 : }
153 : }
154 0 : if (item.request) {
155 : sendRequest(
156 : item.tid, item.oid, item.type, item.member, item.arguments,
157 0 : (item.oid != "UrpProtocolProperties" &&
158 : !item.member.equals(
159 : css::uno::TypeDescription(
160 0 : "com.sun.star.uno.XInterface::release")) &&
161 0 : bridge_->isCurrentContextMode()),
162 0 : item.currentContext);
163 : } else {
164 : sendReply(
165 : item.tid, item.member, item.setter, item.exception,
166 0 : item.returnValue, item.arguments);
167 0 : if (item.setCurrentContextMode) {
168 0 : bridge_->setCurrentContextMode();
169 : }
170 : }
171 0 : }
172 0 : } catch (const css::uno::Exception & e) {
173 : OSL_TRACE(
174 : OSL_LOG_PREFIX "caught UNO exception '%s'",
175 : OUStringToOString(e.Message, RTL_TEXTENCODING_UTF8).getStr());
176 0 : } catch (const std::exception & e) {
177 : OSL_TRACE(OSL_LOG_PREFIX "caught C++ exception '%s'", e.what());
178 : }
179 0 : bridge_->terminate(false);
180 : }
181 :
182 0 : void Writer::sendRequest(
183 : rtl::ByteSequence const & tid, OUString const & oid,
184 : css::uno::TypeDescription const & type,
185 : css::uno::TypeDescription const & member,
186 : std::vector< BinaryAny > const & inArguments, bool currentContextMode,
187 : css::uno::UnoInterfaceReference const & currentContext)
188 : {
189 : OSL_ASSERT(tid.getLength() != 0 && !oid.isEmpty() && member.is());
190 0 : css::uno::TypeDescription t(type);
191 0 : sal_Int32 functionId = 0;
192 0 : bool forceSynchronous = false;
193 0 : member.makeComplete();
194 0 : switch (member.get()->eTypeClass) {
195 : case typelib_TypeClass_INTERFACE_ATTRIBUTE:
196 : {
197 : typelib_InterfaceAttributeTypeDescription * atd =
198 : reinterpret_cast< typelib_InterfaceAttributeTypeDescription * >(
199 0 : member.get());
200 : OSL_ASSERT(atd->pInterface != 0);
201 0 : if (!t.is()) {
202 0 : t = css::uno::TypeDescription(&atd->pInterface->aBase);
203 : }
204 0 : t.makeComplete();
205 : functionId = atd->pInterface->pMapMemberIndexToFunctionIndex[
206 0 : atd->aBase.nPosition];
207 0 : if (!inArguments.empty()) { // setter
208 0 : ++functionId;
209 : }
210 0 : break;
211 : }
212 : case typelib_TypeClass_INTERFACE_METHOD:
213 : {
214 : typelib_InterfaceMethodTypeDescription * mtd =
215 : reinterpret_cast< typelib_InterfaceMethodTypeDescription * >(
216 0 : member.get());
217 : OSL_ASSERT(mtd->pInterface != 0);
218 0 : if (!t.is()) {
219 0 : t = css::uno::TypeDescription(&mtd->pInterface->aBase);
220 : }
221 0 : t.makeComplete();
222 : functionId = mtd->pInterface->pMapMemberIndexToFunctionIndex[
223 0 : mtd->aBase.nPosition];
224 : forceSynchronous = mtd->bOneWay &&
225 0 : functionId != SPECIAL_FUNCTION_ID_RELEASE;
226 0 : break;
227 : }
228 : default:
229 : OSL_ASSERT(false); // this cannot happen
230 0 : break;
231 : }
232 : OSL_ASSERT(functionId >= 0);
233 0 : if (functionId > SAL_MAX_UINT16) {
234 : throw css::uno::RuntimeException(
235 : "function ID too large for URP",
236 0 : css::uno::Reference< css::uno::XInterface >());
237 : }
238 0 : std::vector< unsigned char > buf;
239 0 : bool newType = !(lastType_.is() && t.equals(lastType_));
240 0 : bool newOid = oid != lastOid_;
241 0 : bool newTid = tid != lastTid_;
242 0 : if (newType || newOid || newTid || forceSynchronous || functionId > 0x3FFF)
243 : // > 14 bit function ID
244 : {
245 : Marshal::write8(
246 : &buf,
247 : (0xC0 | (newType ? 0x20 : 0) | (newOid ? 0x10 : 0) |
248 : (newTid ? 0x08 : 0) | (functionId > 0xFF ? 0x04 : 0) |
249 0 : (forceSynchronous ? 0x01 : 0)));
250 : // bit 7: LONGHEADER, bit 6: REQUEST, bit 5: NEWTYPE, bit 4: NEWOID,
251 : // bit 3: NEWTID, bit 2: FUNCTIONID16, bit 0: MOREFLAGS
252 0 : if (forceSynchronous) {
253 0 : Marshal::write8(&buf, 0xC0); // bit 7: MUSTREPLY, bit 6: SYNCHRONOUS
254 : }
255 0 : if (functionId <= 0xFF) {
256 0 : Marshal::write8(&buf, static_cast< sal_uInt8 >(functionId));
257 : } else {
258 0 : Marshal::write16(&buf, static_cast< sal_uInt16 >(functionId));
259 : }
260 0 : if (newType) {
261 0 : marshal_.writeType(&buf, t);
262 : }
263 0 : if (newOid) {
264 0 : marshal_.writeOid(&buf, oid);
265 : }
266 0 : if (newTid) {
267 0 : marshal_.writeTid(&buf, tid);
268 : }
269 0 : } else if (functionId <= 0x3F) { // <= 6 bit function ID
270 0 : Marshal::write8(&buf, static_cast< sal_uInt8 >(functionId));
271 : // bit 7: !LONGHEADER, bit 6: !FUNCTIONID14
272 : } else {
273 : Marshal::write8(
274 0 : &buf, static_cast< sal_uInt8 >(0x40 | (functionId >> 8)));
275 : // bit 7: !LONGHEADER, bit 6: FUNCTIONID14
276 0 : Marshal::write8(&buf, functionId & 0xFF);
277 : }
278 0 : if (currentContextMode) {
279 0 : css::uno::UnoInterfaceReference cc(currentContext);
280 : marshal_.writeValue(
281 : &buf,
282 : css::uno::TypeDescription(
283 : cppu::UnoType<
284 0 : css::uno::Reference< css::uno::XCurrentContext > >::get()),
285 : BinaryAny(
286 : css::uno::TypeDescription(
287 : cppu::UnoType<
288 : css::uno::Reference<
289 0 : css::uno::XCurrentContext > >::get()),
290 0 : &cc.m_pUnoI));
291 : }
292 0 : switch (member.get()->eTypeClass) {
293 : case typelib_TypeClass_INTERFACE_ATTRIBUTE:
294 0 : if (!inArguments.empty()) { // setter
295 : OSL_ASSERT(inArguments.size() == 1);
296 : marshal_.writeValue(
297 : &buf,
298 : css::uno::TypeDescription(
299 : reinterpret_cast<
300 : typelib_InterfaceAttributeTypeDescription * >(
301 0 : member.get())->
302 : pAttributeTypeRef),
303 0 : inArguments.front());
304 : }
305 0 : break;
306 : case typelib_TypeClass_INTERFACE_METHOD:
307 : {
308 : typelib_InterfaceMethodTypeDescription * mtd =
309 : reinterpret_cast< typelib_InterfaceMethodTypeDescription * >(
310 0 : member.get());
311 0 : std::vector< BinaryAny >::const_iterator i(inArguments.begin());
312 0 : for (sal_Int32 j = 0; j != mtd->nParams; ++j) {
313 0 : if (mtd->pParams[j].bIn) {
314 : marshal_.writeValue(
315 : &buf,
316 0 : css::uno::TypeDescription(mtd->pParams[j].pTypeRef),
317 0 : *i++);
318 : }
319 : }
320 : OSL_ASSERT(i == inArguments.end());
321 : break;
322 : }
323 : default:
324 : OSL_ASSERT(false); // this cannot happen
325 0 : break;
326 : }
327 0 : sendMessage(buf);
328 0 : lastType_ = t;
329 0 : lastOid_ = oid;
330 0 : lastTid_ = tid;
331 0 : }
332 :
333 0 : void Writer::sendReply(
334 : rtl::ByteSequence const & tid,
335 : com::sun::star::uno::TypeDescription const & member, bool setter,
336 : bool exception, BinaryAny const & returnValue,
337 : std::vector< BinaryAny > const & outArguments)
338 : {
339 : OSL_ASSERT(tid.getLength() != 0 && member.is() && member.get()->bComplete);
340 0 : std::vector< unsigned char > buf;
341 0 : bool newTid = tid != lastTid_;
342 0 : Marshal::write8(&buf, 0x80 | (exception ? 0x20 : 0) | (newTid ? 0x08 : 0));
343 : // bit 7: LONGHEADER; bit 6: !REQUEST; bit 5: EXCEPTION; bit 3: NEWTID
344 0 : if (newTid) {
345 0 : marshal_.writeTid(&buf, tid);
346 : }
347 0 : if (exception) {
348 : marshal_.writeValue(
349 : &buf,
350 0 : css::uno::TypeDescription(cppu::UnoType< css::uno::Any >::get()),
351 0 : returnValue);
352 : } else {
353 0 : switch (member.get()->eTypeClass) {
354 : case typelib_TypeClass_INTERFACE_ATTRIBUTE:
355 0 : if (!setter) {
356 : marshal_.writeValue(
357 : &buf,
358 : css::uno::TypeDescription(
359 : reinterpret_cast<
360 : typelib_InterfaceAttributeTypeDescription * >(
361 0 : member.get())->
362 : pAttributeTypeRef),
363 0 : returnValue);
364 : }
365 0 : break;
366 : case typelib_TypeClass_INTERFACE_METHOD:
367 : {
368 : typelib_InterfaceMethodTypeDescription * mtd =
369 : reinterpret_cast<
370 : typelib_InterfaceMethodTypeDescription * >(
371 0 : member.get());
372 : marshal_.writeValue(
373 : &buf, css::uno::TypeDescription(mtd->pReturnTypeRef),
374 0 : returnValue);
375 : std::vector< BinaryAny >::const_iterator i(
376 0 : outArguments.begin());
377 0 : for (sal_Int32 j = 0; j != mtd->nParams; ++j) {
378 0 : if (mtd->pParams[j].bOut) {
379 : marshal_.writeValue(
380 : &buf,
381 0 : css::uno::TypeDescription(mtd->pParams[j].pTypeRef),
382 0 : *i++);
383 : }
384 : }
385 : OSL_ASSERT(i == outArguments.end());
386 : break;
387 : }
388 : default:
389 : OSL_ASSERT(false); // this cannot happen
390 0 : break;
391 : }
392 : }
393 0 : sendMessage(buf);
394 0 : lastTid_ = tid;
395 0 : bridge_->decrementCalls();
396 0 : }
397 :
398 0 : void Writer::sendMessage(std::vector< unsigned char > const & buffer) {
399 0 : std::vector< unsigned char > header;
400 0 : if (buffer.size() > SAL_MAX_UINT32) {
401 : throw css::uno::RuntimeException(
402 : "message too large for URP",
403 0 : css::uno::Reference< css::uno::XInterface >());
404 : }
405 0 : Marshal::write32(&header, static_cast< sal_uInt32 >(buffer.size()));
406 0 : Marshal::write32(&header, 1);
407 : OSL_ASSERT(!buffer.empty());
408 0 : unsigned char const * p = &buffer[0];
409 0 : std::vector< unsigned char >::size_type n = buffer.size();
410 : OSL_ASSERT(header.size() <= SAL_MAX_INT32 && SAL_MAX_INT32 <= SAL_MAX_SIZE);
411 0 : sal_Size k = SAL_MAX_INT32 - header.size();
412 0 : if (n < k) {
413 0 : k = static_cast< sal_Size >(n);
414 : }
415 : css::uno::Sequence< sal_Int8 > s(
416 0 : static_cast< sal_Int32 >(header.size() + k));
417 : OSL_ASSERT(!header.empty());
418 : memcpy(
419 0 : s.getArray(), &header[0], static_cast< sal_Size >(header.size()));
420 0 : for (;;) {
421 0 : memcpy(s.getArray() + s.getLength() - k, p, k);
422 : try {
423 0 : bridge_->getConnection()->write(s);
424 0 : } catch (const css::io::IOException & e) {
425 0 : css::uno::Any exc(cppu::getCaughtException());
426 : throw css::lang::WrappedTargetRuntimeException(
427 0 : "Binary URP write raised IO exception: " + e.Message,
428 0 : css::uno::Reference< css::uno::XInterface >(), exc);
429 : }
430 0 : n = static_cast< std::vector< unsigned char >::size_type >(n - k);
431 0 : if (n == 0) {
432 0 : break;
433 : }
434 0 : p += k;
435 0 : k = SAL_MAX_INT32;
436 0 : if (n < k) {
437 0 : k = static_cast< sal_Size >(n);
438 : }
439 0 : s.realloc(k);
440 0 : }
441 0 : }
442 :
443 : }
444 :
445 : /* vim:set shiftwidth=4 softtabstop=4 expandtab: */
|