1*ca5ec200SAndrew Rist /**************************************************************
2cdf0e10cSrcweir  *
3*ca5ec200SAndrew Rist  * Licensed to the Apache Software Foundation (ASF) under one
4*ca5ec200SAndrew Rist  * or more contributor license agreements.  See the NOTICE file
5*ca5ec200SAndrew Rist  * distributed with this work for additional information
6*ca5ec200SAndrew Rist  * regarding copyright ownership.  The ASF licenses this file
7*ca5ec200SAndrew Rist  * to you under the Apache License, Version 2.0 (the
8*ca5ec200SAndrew Rist  * "License"); you may not use this file except in compliance
9*ca5ec200SAndrew Rist  * with the License.  You may obtain a copy of the License at
10*ca5ec200SAndrew Rist  *
11*ca5ec200SAndrew Rist  *   http://www.apache.org/licenses/LICENSE-2.0
12*ca5ec200SAndrew Rist  *
13*ca5ec200SAndrew Rist  * Unless required by applicable law or agreed to in writing,
14*ca5ec200SAndrew Rist  * software distributed under the License is distributed on an
15*ca5ec200SAndrew Rist  * "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
16*ca5ec200SAndrew Rist  * KIND, either express or implied.  See the License for the
17*ca5ec200SAndrew Rist  * specific language governing permissions and limitations
18*ca5ec200SAndrew Rist  * under the License.
19*ca5ec200SAndrew Rist  *
20*ca5ec200SAndrew Rist  *************************************************************/
21*ca5ec200SAndrew Rist 
22*ca5ec200SAndrew Rist 
23cdf0e10cSrcweir 
24cdf0e10cSrcweir #include "oox/helper/binaryoutputstream.hxx"
25cdf0e10cSrcweir 
26cdf0e10cSrcweir #include <com/sun/star/io/XOutputStream.hpp>
27cdf0e10cSrcweir #include <com/sun/star/io/XSeekable.hpp>
28cdf0e10cSrcweir #include <osl/diagnose.h>
29cdf0e10cSrcweir #include <string.h>
30cdf0e10cSrcweir 
31cdf0e10cSrcweir namespace oox {
32cdf0e10cSrcweir 
33cdf0e10cSrcweir // ============================================================================
34cdf0e10cSrcweir 
35cdf0e10cSrcweir using namespace ::com::sun::star::io;
36cdf0e10cSrcweir using namespace ::com::sun::star::uno;
37cdf0e10cSrcweir 
38cdf0e10cSrcweir namespace {
39cdf0e10cSrcweir 
40cdf0e10cSrcweir const sal_Int32 OUTPUTSTREAM_BUFFERSIZE     = 0x8000;
41cdf0e10cSrcweir 
42cdf0e10cSrcweir } // namespace
43cdf0e10cSrcweir 
44cdf0e10cSrcweir // ============================================================================
45cdf0e10cSrcweir 
BinaryXOutputStream(const Reference<XOutputStream> & rxOutStrm,bool bAutoClose)46cdf0e10cSrcweir BinaryXOutputStream::BinaryXOutputStream( const Reference< XOutputStream >& rxOutStrm, bool bAutoClose ) :
47cdf0e10cSrcweir     BinaryStreamBase( Reference< XSeekable >( rxOutStrm, UNO_QUERY ).is() ),
48cdf0e10cSrcweir     BinaryXSeekableStream( Reference< XSeekable >( rxOutStrm, UNO_QUERY ) ),
49cdf0e10cSrcweir     maBuffer( OUTPUTSTREAM_BUFFERSIZE ),
50cdf0e10cSrcweir     mxOutStrm( rxOutStrm ),
51cdf0e10cSrcweir     mbAutoClose( bAutoClose && rxOutStrm.is() )
52cdf0e10cSrcweir {
53cdf0e10cSrcweir     mbEof = !mxOutStrm.is();
54cdf0e10cSrcweir }
55cdf0e10cSrcweir 
~BinaryXOutputStream()56cdf0e10cSrcweir BinaryXOutputStream::~BinaryXOutputStream()
57cdf0e10cSrcweir {
58cdf0e10cSrcweir     close();
59cdf0e10cSrcweir }
60cdf0e10cSrcweir 
close()61cdf0e10cSrcweir void BinaryXOutputStream::close()
62cdf0e10cSrcweir {
63cdf0e10cSrcweir     OSL_ENSURE( !mbAutoClose || mxOutStrm.is(), "BinaryXOutputStream::close - invalid call" );
64cdf0e10cSrcweir     if( mxOutStrm.is() ) try
65cdf0e10cSrcweir     {
66cdf0e10cSrcweir         mxOutStrm->flush();
67cdf0e10cSrcweir         if( mbAutoClose )
68cdf0e10cSrcweir             mxOutStrm->closeOutput();
69cdf0e10cSrcweir     }
70cdf0e10cSrcweir     catch( Exception& )
71cdf0e10cSrcweir     {
72cdf0e10cSrcweir         OSL_ENSURE( false, "BinaryXOutputStream::close - closing output stream failed" );
73cdf0e10cSrcweir     }
74cdf0e10cSrcweir     mxOutStrm.clear();
75cdf0e10cSrcweir     mbAutoClose = false;
76cdf0e10cSrcweir     BinaryXSeekableStream::close();
77cdf0e10cSrcweir }
78cdf0e10cSrcweir 
writeData(const StreamDataSequence & rData,size_t)79cdf0e10cSrcweir void BinaryXOutputStream::writeData( const StreamDataSequence& rData, size_t /*nAtomSize*/ )
80cdf0e10cSrcweir {
81cdf0e10cSrcweir     if( mxOutStrm.is() ) try
82cdf0e10cSrcweir     {
83cdf0e10cSrcweir         mxOutStrm->writeBytes( rData );
84cdf0e10cSrcweir     }
85cdf0e10cSrcweir     catch( Exception& )
86cdf0e10cSrcweir     {
87cdf0e10cSrcweir         OSL_ENSURE( false, "BinaryXOutputStream::writeData - stream read error" );
88cdf0e10cSrcweir     }
89cdf0e10cSrcweir }
90cdf0e10cSrcweir 
writeMemory(const void * pMem,sal_Int32 nBytes,size_t nAtomSize)91cdf0e10cSrcweir void BinaryXOutputStream::writeMemory( const void* pMem, sal_Int32 nBytes, size_t nAtomSize )
92cdf0e10cSrcweir {
93cdf0e10cSrcweir     if( mxOutStrm.is() && (nBytes > 0) )
94cdf0e10cSrcweir     {
95cdf0e10cSrcweir         sal_Int32 nBufferSize = getLimitedValue< sal_Int32, sal_Int32 >( nBytes, 0, (OUTPUTSTREAM_BUFFERSIZE / nAtomSize) * nAtomSize );
96cdf0e10cSrcweir         const sal_uInt8* pnMem = reinterpret_cast< const sal_uInt8* >( pMem );
97cdf0e10cSrcweir         while( nBytes > 0 )
98cdf0e10cSrcweir         {
99cdf0e10cSrcweir             sal_Int32 nWriteSize = getLimitedValue< sal_Int32, sal_Int32 >( nBytes, 0, nBufferSize );
100cdf0e10cSrcweir             maBuffer.realloc( nWriteSize );
101cdf0e10cSrcweir             memcpy( maBuffer.getArray(), pnMem, static_cast< size_t >( nWriteSize ) );
102cdf0e10cSrcweir             writeData( maBuffer, nAtomSize );
103cdf0e10cSrcweir             pnMem += nWriteSize;
104cdf0e10cSrcweir             nBytes -= nWriteSize;
105cdf0e10cSrcweir         }
106cdf0e10cSrcweir     }
107cdf0e10cSrcweir }
108cdf0e10cSrcweir 
109cdf0e10cSrcweir // ============================================================================
110cdf0e10cSrcweir 
SequenceOutputStream(StreamDataSequence & rData)111cdf0e10cSrcweir SequenceOutputStream::SequenceOutputStream( StreamDataSequence& rData ) :
112cdf0e10cSrcweir     BinaryStreamBase( true ),
113cdf0e10cSrcweir     SequenceSeekableStream( rData )
114cdf0e10cSrcweir {
115cdf0e10cSrcweir }
116cdf0e10cSrcweir 
writeData(const StreamDataSequence & rData,size_t nAtomSize)117cdf0e10cSrcweir void SequenceOutputStream::writeData( const StreamDataSequence& rData, size_t nAtomSize )
118cdf0e10cSrcweir {
119cdf0e10cSrcweir     if( mpData && rData.hasElements() )
120cdf0e10cSrcweir         writeMemory( rData.getConstArray(), rData.getLength(), nAtomSize );
121cdf0e10cSrcweir }
122cdf0e10cSrcweir 
writeMemory(const void * pMem,sal_Int32 nBytes,size_t)123cdf0e10cSrcweir void SequenceOutputStream::writeMemory( const void* pMem, sal_Int32 nBytes, size_t /*nAtomSize*/ )
124cdf0e10cSrcweir {
125cdf0e10cSrcweir     if( mpData && (nBytes > 0) )
126cdf0e10cSrcweir     {
127cdf0e10cSrcweir         if( mpData->getLength() - mnPos < nBytes )
128cdf0e10cSrcweir             const_cast< StreamDataSequence* >( mpData )->realloc( mnPos + nBytes );
129cdf0e10cSrcweir         memcpy( const_cast< StreamDataSequence* >( mpData )->getArray() + mnPos, pMem, static_cast< size_t >( nBytes ) );
130cdf0e10cSrcweir         mnPos += nBytes;
131cdf0e10cSrcweir     }
132cdf0e10cSrcweir }
133cdf0e10cSrcweir 
134cdf0e10cSrcweir // ============================================================================
135cdf0e10cSrcweir 
136cdf0e10cSrcweir } // namespace oox
137cdf0e10cSrcweir 
138