1 /* -*- Mode: C++; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4 -*- */
3 * This file is part of the LibreOffice project.
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/.
9 * This file incorporates work covered by the following license notice:
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 .
20 #include <sal/config.h>
22 #include <com/sun/star/io/BufferSizeExceededException.hpp>
23 #include <com/sun/star/io/NotConnectedException.hpp>
24 #include <com/sun/star/io/XPipe.hpp>
25 #include <com/sun/star/io/XConnectable.hpp>
27 #include <com/sun/star/lang/XServiceInfo.hpp>
29 #include <cppuhelper/implbase.hxx>
30 #include <cppuhelper/supportsservice.hxx>
32 #include <osl/conditn.hxx>
33 #include <osl/mutex.hxx>
39 using namespace ::osl
;
40 using namespace ::cppu
;
41 using namespace ::com::sun::star::uno
;
42 using namespace ::com::sun::star::io
;
43 using namespace ::com::sun::star::lang
;
45 #include <services.hxx>
46 #include "streamhelper.hxx"
48 namespace com::sun::star::uno
{ class XComponentContext
; }
50 // Implementation and service names
51 #define IMPLEMENTATION_NAME "com.sun.star.comp.io.stm.Pipe"
56 public WeakImplHelper
< XPipe
, XConnectable
, XServiceInfo
>
61 public: // XInputStream
62 virtual sal_Int32 SAL_CALL
readBytes(Sequence
< sal_Int8
>& aData
, sal_Int32 nBytesToRead
) override
;
63 virtual sal_Int32 SAL_CALL
readSomeBytes(Sequence
< sal_Int8
>& aData
, sal_Int32 nMaxBytesToRead
) override
;
64 virtual void SAL_CALL
skipBytes(sal_Int32 nBytesToSkip
) override
;
65 virtual sal_Int32 SAL_CALL
available() override
;
66 virtual void SAL_CALL
closeInput() override
;
68 public: // XOutputStream
70 virtual void SAL_CALL
writeBytes(const Sequence
< sal_Int8
>& aData
) override
;
71 virtual void SAL_CALL
flush() override
;
72 virtual void SAL_CALL
closeOutput() override
;
74 public: // XConnectable
75 virtual void SAL_CALL
setPredecessor(const Reference
< XConnectable
>& aPredecessor
) override
;
76 virtual Reference
< XConnectable
> SAL_CALL
getPredecessor() override
;
77 virtual void SAL_CALL
setSuccessor(const Reference
< XConnectable
> & aSuccessor
) override
;
78 virtual Reference
< XConnectable
> SAL_CALL
getSuccessor() override
;
81 public: // XServiceInfo
82 OUString SAL_CALL
getImplementationName() override
;
83 Sequence
< OUString
> SAL_CALL
getSupportedServiceNames() override
;
84 sal_Bool SAL_CALL
supportsService(const OUString
& ServiceName
) override
;
88 Reference
< XConnectable
> m_succ
;
89 Reference
< XConnectable
> m_pred
;
91 sal_Int32 m_nBytesToSkip
;
93 bool m_bOutputStreamClosed
;
94 bool m_bInputStreamClosed
;
96 osl::Condition m_conditionBytesAvail
;
98 std::unique_ptr
<MemFIFO
> m_pFIFO
;
102 OPipeImpl::OPipeImpl()
104 , m_bOutputStreamClosed(false )
105 , m_bInputStreamClosed( false )
106 , m_pFIFO( new MemFIFO
)
112 sal_Int32
OPipeImpl::readBytes(Sequence
< sal_Int8
>& aData
, sal_Int32 nBytesToRead
)
116 { // start guarded section
117 MutexGuard
guard( m_mutexAccess
);
118 if( m_bInputStreamClosed
)
120 throw NotConnectedException(
121 "Pipe::readBytes NotConnectedException",
124 sal_Int32 nOccupiedBufferLen
= m_pFIFO
->getSize();
126 if( m_bOutputStreamClosed
&& nBytesToRead
> nOccupiedBufferLen
)
128 nBytesToRead
= nOccupiedBufferLen
;
131 if( nOccupiedBufferLen
< nBytesToRead
)
133 // wait outside guarded section
134 m_conditionBytesAvail
.reset();
137 // necessary bytes are available
138 m_pFIFO
->read( aData
, nBytesToRead
);
141 } // end guarded section
143 // wait for new data outside guarded section!
144 m_conditionBytesAvail
.wait();
149 sal_Int32
OPipeImpl::readSomeBytes(Sequence
< sal_Int8
>& aData
, sal_Int32 nMaxBytesToRead
)
153 MutexGuard
guard( m_mutexAccess
);
154 if( m_bInputStreamClosed
)
156 throw NotConnectedException(
157 "Pipe::readSomeBytes NotConnectedException",
160 if( m_pFIFO
->getSize() )
162 sal_Int32 nSize
= std::min( nMaxBytesToRead
, m_pFIFO
->getSize() );
163 aData
.realloc( nSize
);
164 m_pFIFO
->read( aData
, nSize
);
168 if( m_bOutputStreamClosed
)
170 // no bytes in buffer anymore
175 m_conditionBytesAvail
.wait();
180 void OPipeImpl::skipBytes(sal_Int32 nBytesToSkip
)
182 MutexGuard
guard( m_mutexAccess
);
183 if( m_bInputStreamClosed
)
185 throw NotConnectedException(
186 "Pipe::skipBytes NotConnectedException",
192 > std::numeric_limits
< sal_Int32
>::max() - m_nBytesToSkip
) )
194 throw BufferSizeExceededException(
195 "Pipe::skipBytes BufferSizeExceededException",
198 m_nBytesToSkip
+= nBytesToSkip
;
200 nBytesToSkip
= std::min( m_pFIFO
->getSize() , m_nBytesToSkip
);
201 m_pFIFO
->skip( nBytesToSkip
);
202 m_nBytesToSkip
-= nBytesToSkip
;
206 sal_Int32
OPipeImpl::available()
208 MutexGuard
guard( m_mutexAccess
);
209 if( m_bInputStreamClosed
)
211 throw NotConnectedException(
212 "Pipe::available NotConnectedException",
215 return m_pFIFO
->getSize();
218 void OPipeImpl::closeInput()
220 MutexGuard
guard( m_mutexAccess
);
222 m_bInputStreamClosed
= true;
226 // readBytes may throw an exception
227 m_conditionBytesAvail
.set();
229 setSuccessor( Reference
< XConnectable
> () );
233 void OPipeImpl::writeBytes(const Sequence
< sal_Int8
>& aData
)
235 MutexGuard
guard( m_mutexAccess
);
237 if( m_bOutputStreamClosed
)
239 throw NotConnectedException(
240 "Pipe::writeBytes NotConnectedException (outputstream)",
244 if( m_bInputStreamClosed
)
246 throw NotConnectedException(
247 "Pipe::writeBytes NotConnectedException (inputstream)",
252 sal_Int32 nLen
= aData
.getLength();
253 if( m_nBytesToSkip
&& m_nBytesToSkip
>= nLen
) {
254 // all must be skipped - forget whole call
255 m_nBytesToSkip
-= nLen
;
259 // adjust buffersize if necessary
262 Sequence
< sal_Int8
> seqCopy( nLen
- m_nBytesToSkip
);
263 memcpy( seqCopy
.getArray() , &( aData
.getConstArray()[m_nBytesToSkip
] ) , nLen
-m_nBytesToSkip
);
264 m_pFIFO
->write( seqCopy
);
268 m_pFIFO
->write( aData
);
272 // readBytes may check again if enough bytes are available
273 m_conditionBytesAvail
.set();
277 void OPipeImpl::flush()
279 // nothing to do for a pipe
282 void OPipeImpl::closeOutput()
284 MutexGuard
guard( m_mutexAccess
);
286 m_bOutputStreamClosed
= true;
287 m_conditionBytesAvail
.set();
288 setPredecessor( Reference
< XConnectable
> () );
292 void OPipeImpl::setSuccessor( const Reference
< XConnectable
> &r
)
294 /// if the references match, nothing needs to be done
296 /// store the reference for later use
301 m_succ
->setPredecessor(
302 Reference
< XConnectable
> ( static_cast< XConnectable
* >(this) ) );
307 Reference
< XConnectable
> OPipeImpl::getSuccessor()
314 void OPipeImpl::setPredecessor( const Reference
< XConnectable
> &r
)
319 m_pred
->setSuccessor(
320 Reference
< XConnectable
> ( static_cast< XConnectable
* >(this) ) );
325 Reference
< XConnectable
> OPipeImpl::getPredecessor()
332 OUString
OPipeImpl::getImplementationName()
334 return OPipeImpl_getImplementationName();
338 sal_Bool
OPipeImpl::supportsService(const OUString
& ServiceName
)
340 return cppu::supportsService(this, ServiceName
);
344 Sequence
< OUString
> OPipeImpl::getSupportedServiceNames()
346 return OPipeImpl_getSupportedServiceNames();
349 /* implementation functions
355 Reference
< XInterface
> OPipeImpl_CreateInstance(
356 SAL_UNUSED_PARAMETER
const Reference
< XComponentContext
> & )
358 OPipeImpl
*p
= new OPipeImpl
;
360 return Reference
< XInterface
> ( static_cast< OWeakObject
* >(p
) );
364 OUString
OPipeImpl_getImplementationName()
366 return IMPLEMENTATION_NAME
;
369 Sequence
<OUString
> OPipeImpl_getSupportedServiceNames()
371 Sequence
<OUString
> aRet
{ "com.sun.star.io.Pipe" };
377 /* vim:set shiftwidth=4 softtabstop=4 expandtab: */