Files
libreoffice/io/source/stm/opump.cxx

464 lines
12 KiB
C++
Raw Normal View History

/* -*- Mode: C++; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4 -*- */
2012-07-03 13:27:15 +01:00
/*
* This file is part of the LibreOffice project.
*
* This Source Code Form is subject to the terms of the Mozilla Public
* License, v. 2.0. If a copy of the MPL was not distributed with this
* file, You can obtain one at http://mozilla.org/MPL/2.0/.
*
* This file incorporates work covered by the following license notice:
*
* Licensed to the Apache Software Foundation (ASF) under one or more
* contributor license agreements. See the NOTICE file distributed
* with this work for additional information regarding copyright
* ownership. The ASF licenses this file to you under the Apache
* License, Version 2.0 (the "License"); you may not use this file
* except in compliance with the License. You may obtain a copy of
* the License at http://www.apache.org/licenses/LICENSE-2.0 .
*/
2000-09-18 16:24:28 +00:00
2000-09-18 16:24:28 +00:00
#include <stdio.h>
#include <sal/log.hxx>
2000-09-18 16:24:28 +00:00
#include <com/sun/star/io/IOException.hpp>
#include <com/sun/star/io/NotConnectedException.hpp>
2000-09-18 16:24:28 +00:00
#include <com/sun/star/io/XActiveDataSource.hpp>
#include <com/sun/star/io/XActiveDataSink.hpp>
#include <com/sun/star/io/XActiveDataControl.hpp>
#include <com/sun/star/io/XConnectable.hpp>
#include <com/sun/star/lang/XSingleServiceFactory.hpp>
#include <com/sun/star/lang/XMultiServiceFactory.hpp>
2001-03-20 08:04:37 +00:00
#include <com/sun/star/lang/XServiceInfo.hpp>
2000-09-18 16:24:28 +00:00
#include <com/sun/star/registry/XRegistryKey.hpp>
#include <uno/dispatcher.h>
#include <uno/mapping.hxx>
#include <cppuhelper/implbase.hxx>
2000-09-18 16:24:28 +00:00
#include <cppuhelper/factory.hxx>
#include <cppuhelper/interfacecontainer.hxx>
#include <cppuhelper/supportsservice.hxx>
2000-09-18 16:24:28 +00:00
#include <osl/mutex.hxx>
#include <osl/thread.h>
2001-03-20 08:04:37 +00:00
2000-09-18 16:24:28 +00:00
using namespace osl;
using namespace std;
using namespace cppu;
using namespace com::sun::star::uno;
using namespace com::sun::star::lang;
using namespace com::sun::star::registry;
using namespace com::sun::star::io;
#include <services.hxx>
2001-03-20 08:04:37 +00:00
2000-09-18 16:24:28 +00:00
namespace io_stm {
class Pump : public WeakImplHelper<
2001-03-20 08:04:37 +00:00
XActiveDataSource, XActiveDataSink, XActiveDataControl, XConnectable, XServiceInfo >
2000-09-18 16:24:28 +00:00
{
Mutex m_aMutex;
oslThread m_aThread;
Reference< XConnectable > m_xPred;
Reference< XConnectable > m_xSucc;
Reference< XInputStream > m_xInput;
Reference< XOutputStream > m_xOutput;
OInterfaceContainerHelper m_cnt;
bool m_closeFired;
2000-09-18 16:24:28 +00:00
void run();
static void static_run( void* pObject );
void close();
void fireClose();
void fireStarted();
void fireTerminated();
void fireError( const Any &a );
2000-09-18 16:24:28 +00:00
public:
Pump();
virtual ~Pump() override;
2000-09-18 16:24:28 +00:00
// XActiveDataSource
virtual void SAL_CALL setOutputStream( const Reference< css::io::XOutputStream >& xOutput ) override;
virtual Reference< css::io::XOutputStream > SAL_CALL getOutputStream() override;
2000-09-18 16:24:28 +00:00
// XActiveDataSink
virtual void SAL_CALL setInputStream( const Reference< css::io::XInputStream >& xStream ) override;
virtual Reference< css::io::XInputStream > SAL_CALL getInputStream() override;
2000-09-18 16:24:28 +00:00
// XActiveDataControl
virtual void SAL_CALL addListener( const Reference< css::io::XStreamListener >& xListener ) override;
virtual void SAL_CALL removeListener( const Reference< css::io::XStreamListener >& xListener ) override;
virtual void SAL_CALL start() override;
virtual void SAL_CALL terminate() override;
2000-09-18 16:24:28 +00:00
// XConnectable
virtual void SAL_CALL setPredecessor( const Reference< css::io::XConnectable >& xPred ) override;
virtual Reference< css::io::XConnectable > SAL_CALL getPredecessor() override;
virtual void SAL_CALL setSuccessor( const Reference< css::io::XConnectable >& xSucc ) override;
virtual Reference< css::io::XConnectable > SAL_CALL getSuccessor() override;
2000-09-18 16:24:28 +00:00
2001-03-20 08:04:37 +00:00
public: // XServiceInfo
virtual OUString SAL_CALL getImplementationName() override;
virtual Sequence< OUString > SAL_CALL getSupportedServiceNames() override;
virtual sal_Bool SAL_CALL supportsService(const OUString& ServiceName) override;
2000-09-18 16:24:28 +00:00
};
Pump::Pump() : m_aThread( nullptr ),
m_cnt( m_aMutex ),
m_closeFired( false )
2000-09-18 16:24:28 +00:00
{
}
Pump::~Pump()
{
// exit gracefully
if( m_aThread )
{
osl_joinWithThread( m_aThread );
osl_destroyThread( m_aThread );
}
2000-09-18 16:24:28 +00:00
}
void Pump::fireError( const Any & exception )
{
OInterfaceIteratorHelper iter( m_cnt );
while( iter.hasMoreElements() )
{
try
{
static_cast< XStreamListener * > ( iter.next() )->error( exception );
}
2011-12-15 13:51:29 +09:00
catch ( const RuntimeException &e )
{
SAL_WARN("io.streams","com.sun.star.comp.stoc.Pump: unexpected exception during calling listeners" << e);
}
}
}
void Pump::fireClose()
{
bool bFire = false;
{
MutexGuard guard( m_aMutex );
if( ! m_closeFired )
{
m_closeFired = true;
bFire = true;
}
}
if( bFire )
{
OInterfaceIteratorHelper iter( m_cnt );
while( iter.hasMoreElements() )
{
try
{
static_cast< XStreamListener * > ( iter.next() )->closed( );
}
2011-12-15 13:51:29 +09:00
catch ( const RuntimeException &e )
{
SAL_WARN("io.streams","com.sun.star.comp.stoc.Pump: unexpected exception during calling listeners" << e);
}
}
}
}
void Pump::fireStarted()
2000-09-18 16:24:28 +00:00
{
OInterfaceIteratorHelper iter( m_cnt );
while( iter.hasMoreElements() )
2000-09-18 16:24:28 +00:00
{
try
{
static_cast< XStreamListener * > ( iter.next() )->started( );
2000-09-18 16:24:28 +00:00
}
2011-12-15 13:51:29 +09:00
catch ( const RuntimeException &e )
2000-09-18 16:24:28 +00:00
{
SAL_WARN("io.streams","com.sun.star.comp.stoc.Pump: unexpected exception during calling listeners" << e);
2000-09-18 16:24:28 +00:00
}
}
}
void Pump::fireTerminated()
{
OInterfaceIteratorHelper iter( m_cnt );
while( iter.hasMoreElements() )
{
try
{
static_cast< XStreamListener * > ( iter.next() )->terminated();
}
2011-12-15 13:51:29 +09:00
catch ( const RuntimeException &e )
{
SAL_WARN("io.streams","com.sun.star.comp.stoc.Pump: unexpected exception during calling listeners" << e);
}
}
}
2000-09-18 16:24:28 +00:00
void Pump::close()
{
// close streams and release references
Reference< XInputStream > rInput;
Reference< XOutputStream > rOutput;
{
MutexGuard guard( m_aMutex );
rInput = m_xInput;
m_xInput.clear();
rOutput = m_xOutput;
m_xOutput.clear();
m_xSucc.clear();
m_xPred.clear();
}
if( rInput.is() )
2000-09-18 16:24:28 +00:00
{
try
{
rInput->closeInput();
}
catch( Exception & )
{
// go down calm
}
2000-09-18 16:24:28 +00:00
}
if( rOutput.is() )
2000-09-18 16:24:28 +00:00
{
try
{
rOutput->closeOutput();
}
catch( Exception & )
{
// go down calm
}
2000-09-18 16:24:28 +00:00
}
}
void Pump::static_run( void* pObject )
{
osl_setThreadName("io_stm::Pump::run()");
static_cast<Pump*>(pObject)->run();
static_cast<Pump*>(pObject)->release();
2000-09-18 16:24:28 +00:00
}
void Pump::run()
{
try
{
fireStarted();
2000-09-18 16:24:28 +00:00
try
{
Reference< XInputStream > rInput;
Reference< XOutputStream > rOutput;
{
Guard< Mutex > aGuard( m_aMutex );
rInput = m_xInput;
rOutput = m_xOutput;
}
if( ! rInput.is() )
{
throw NotConnectedException( "no input stream set", static_cast<OWeakObject*>(this) );
}
2000-09-18 16:24:28 +00:00
Sequence< sal_Int8 > aData;
while( rInput->readSomeBytes( aData, 65536 ) )
2000-09-18 16:24:28 +00:00
{
if( ! rOutput.is() )
{
throw NotConnectedException( "no output stream set", static_cast<OWeakObject*>(this) );
}
rOutput->writeBytes( aData );
2000-09-18 16:24:28 +00:00
osl_yieldThread();
}
}
2011-12-15 13:51:29 +09:00
catch ( const IOException & e )
2000-09-18 16:24:28 +00:00
{
fireError( makeAny( e ) );
2000-09-18 16:24:28 +00:00
}
2011-12-15 13:51:29 +09:00
catch ( const RuntimeException & e )
2000-09-18 16:24:28 +00:00
{
fireError( makeAny( e ) );
2000-09-18 16:24:28 +00:00
}
2011-12-15 13:51:29 +09:00
catch ( const Exception & e )
2000-09-18 16:24:28 +00:00
{
fireError( makeAny( e ) );
2000-09-18 16:24:28 +00:00
}
close();
fireClose();
2000-09-18 16:24:28 +00:00
}
catch ( const css::uno::Exception &e )
2000-09-18 16:24:28 +00:00
{
// we are the last on the stack.
// this is to avoid crashing the program, when e.g. a bridge crashes
SAL_WARN("io.streams","com.sun.star.comp.stoc.Pump: unexpected exception during calling listeners" << e);
2000-09-18 16:24:28 +00:00
}
}
2000-09-18 16:24:28 +00:00
/*
* XConnectable
*/
void Pump::setPredecessor( const Reference< XConnectable >& xPred )
2000-09-18 16:24:28 +00:00
{
Guard< Mutex > aGuard( m_aMutex );
m_xPred = xPred;
}
Reference< XConnectable > Pump::getPredecessor()
2000-09-18 16:24:28 +00:00
{
Guard< Mutex > aGuard( m_aMutex );
return m_xPred;
}
void Pump::setSuccessor( const Reference< XConnectable >& xSucc )
2000-09-18 16:24:28 +00:00
{
Guard< Mutex > aGuard( m_aMutex );
m_xSucc = xSucc;
}
Reference< XConnectable > Pump::getSuccessor()
2000-09-18 16:24:28 +00:00
{
Guard< Mutex > aGuard( m_aMutex );
return m_xSucc;
}
2000-09-18 16:24:28 +00:00
/*
* XActiveDataControl
*/
void Pump::addListener( const Reference< XStreamListener >& xListener )
2000-09-18 16:24:28 +00:00
{
m_cnt.addInterface( xListener );
2000-09-18 16:24:28 +00:00
}
void Pump::removeListener( const Reference< XStreamListener >& xListener )
2000-09-18 16:24:28 +00:00
{
m_cnt.removeInterface( xListener );
2000-09-18 16:24:28 +00:00
}
void Pump::start()
2000-09-18 16:24:28 +00:00
{
Guard< Mutex > aGuard( m_aMutex );
m_aThread = osl_createSuspendedThread(Pump::static_run,this);
if( !m_aThread )
{
throw RuntimeException(
"Pump::start Couldn't create worker thread",
*this);
}
// will be released by OPump::static_run
acquire();
osl_resumeThread( m_aThread );
2000-09-18 16:24:28 +00:00
}
void Pump::terminate()
2000-09-18 16:24:28 +00:00
{
close();
// wait for the worker to die
if( m_aThread )
osl_joinWithThread( m_aThread );
2000-09-18 16:24:28 +00:00
fireTerminated();
fireClose();
2000-09-18 16:24:28 +00:00
}
2000-09-18 16:24:28 +00:00
/*
* XActiveDataSink
*/
void Pump::setInputStream( const Reference< XInputStream >& xStream )
2000-09-18 16:24:28 +00:00
{
Guard< Mutex > aGuard( m_aMutex );
m_xInput = xStream;
Reference< XConnectable > xConnect( xStream, UNO_QUERY );
if( xConnect.is() )
xConnect->setSuccessor( this );
// data transfer starts in XActiveDataControl::start
}
Reference< XInputStream > Pump::getInputStream()
2000-09-18 16:24:28 +00:00
{
Guard< Mutex > aGuard( m_aMutex );
return m_xInput;
}
2000-09-18 16:24:28 +00:00
/*
* XActiveDataSource
*/
void Pump::setOutputStream( const Reference< XOutputStream >& xOut )
2000-09-18 16:24:28 +00:00
{
Guard< Mutex > aGuard( m_aMutex );
m_xOutput = xOut;
Reference< XConnectable > xConnect( xOut, UNO_QUERY );
if( xConnect.is() )
xConnect->setPredecessor( this );
// data transfer starts in XActiveDataControl::start
}
Reference< XOutputStream > Pump::getOutputStream()
2000-09-18 16:24:28 +00:00
{
Guard< Mutex > aGuard( m_aMutex );
return m_xOutput;
}
2001-03-20 08:04:37 +00:00
// XServiceInfo
OUString Pump::getImplementationName()
2001-03-20 08:04:37 +00:00
{
return OPumpImpl_getImplementationName();
}
// XServiceInfo
sal_Bool Pump::supportsService(const OUString& ServiceName)
2001-03-20 08:04:37 +00:00
{
return cppu::supportsService(this, ServiceName);
2001-03-20 08:04:37 +00:00
}
// XServiceInfo
Sequence< OUString > Pump::getSupportedServiceNames()
2001-03-20 08:04:37 +00:00
{
return OPumpImpl_getSupportedServiceNames();
}
Reference< XInterface > SAL_CALL OPumpImpl_CreateInstance(
SAL_UNUSED_PARAMETER const Reference< XComponentContext > & )
2000-09-18 16:24:28 +00:00
{
return Reference< XInterface >( *new Pump );
}
OUString OPumpImpl_getImplementationName()
{
return OUString("com.sun.star.comp.io.Pump");
2000-09-18 16:24:28 +00:00
}
Sequence<OUString> OPumpImpl_getSupportedServiceNames()
2000-09-18 16:24:28 +00:00
{
return Sequence< OUString > { "com.sun.star.io.Pump" };
2000-09-18 16:24:28 +00:00
}
}
/* vim:set shiftwidth=4 softtabstop=4 expandtab: */