/* This file is part of the KDE project Copyright (C) 2003 Arnold Krille <arnold@arnoldarts.de> This library is free software; you can redistribute it and/or modify it under the terms of the GNU Library General Public License version 2 as published by the Free Software Foundation. This library is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU Library General Public License for more details. You should have received a copy of the GNU Library General Public License along with this library; see the file COPYING.LIB. If not, write to the Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA. */ #include "kaudioplaystream.h" #include "kaudioplaystream_p.h" #include <kartsserver.h> #include <kaudiomanagerplay.h> #include <artsflow.h> #include <soundserver.h> #include <tdeglobal.h> #include <kdebug.h> #include <tqstring.h> //#include <tqptrqueue.h> //#include <tqcstring.h> //QByteArray #include <string.h> // for strncpy //#include <assert.h> KAudioPlayStreamPrivate::KAudioPlayStreamPrivate( KArtsServer* server, const TQString title, TQObject* p, const char* n ) : TQObject( p,n ) , _server( server ) , _play( new KAudioManagerPlay( _server, title ) ) , _effectrack( Arts::StereoEffectStack::null() ) , _polling( true ), _attached( false ), _effects( true ) { kdDebug( 400 ) << k_funcinfo << endl; initaRts(); } KAudioPlayStreamPrivate::~KAudioPlayStreamPrivate() { kdDebug( 400 ) << k_funcinfo << endl; _play->stop(); if ( _effects ) _effectrack.stop(); _bs2a.stop(); } void KAudioPlayStreamPrivate::initaRts() { kdDebug( 400 ) << k_funcinfo << endl; _effectrack = Arts::DynamicCast( _server->server().createObject( "Arts::StereoEffectStack" ) ); if ( _effectrack.isNull() ) { kdWarning( 400 ) << "Couldn't create EffectStack!" << endl; _effects = false; } _bs2a = Arts::DynamicCast( _server->server().createObject( "Arts::ByteStreamToAudio" ) ); if ( _bs2a.isNull() ) kdFatal( 400 ) << "Couldn't create ByteStreamToAudio" << endl; if ( _effects ) { Arts::connect( _effectrack, _play->amanPlay() ); Arts::connect( _bs2a, _effectrack ); } else { Arts::connect( _bs2a, _play->amanPlay() ); } _play->start(); if ( _effects ) _effectrack.start(); } KAudioPlayStream::KAudioPlayStream( KArtsServer* server, const TQString title, TQObject* p, const char* n ) : TQObject( p,n ) , d( new KAudioPlayStreamPrivate( server, title, this ) ) { kdDebug( 400 ) << k_funcinfo << endl; } KAudioPlayStream::~KAudioPlayStream() { kdDebug( 400 ) << k_funcinfo << endl; } void KAudioPlayStream::setPolling( bool n ) { d->_polling = n; } bool KAudioPlayStream::polling() const { return d->_polling; } bool KAudioPlayStream::running() const { return d->_attached; } Arts::StereoEffectStack KAudioPlayStream::effectStack() const { return d->_effectrack; } void KAudioPlayStream::start( int samplingRate, int bits, int channels ) { kdDebug( 400 ) << k_funcinfo << "samplingRate: " << samplingRate << " bits: " << bits << " channels: " << channels << endl; if ( !d->_attached ) { d->_bs2a.samplingRate( samplingRate ); d->_bs2a.channels( channels ); d->_bs2a.bits( bits ); d->_sender = new KByteSoundProducer( this, d->_server->server().minStreamBufferTime(), samplingRate, bits, channels, "PS" ); d->_artssender = Arts::ByteSoundProducerV2::_from_base( d->_sender ); Arts::connect( d->_artssender, "outdata", d->_bs2a, "indata" ); d->_bs2a.start(); d->_artssender.start(); // // Needed? Arts::Dispatcher::the()->ioManager()->processOneEvent( false ); d->_attached = true; emit running( d->_attached ); } } void KAudioPlayStream::stop() { kdDebug( 400 ) << k_funcinfo << endl; if ( d->_attached ) { d->_attached = false; d->_bs2a.stop(); d->_artssender.stop(); // Shortly stop the play so we dont get clicks and artefacts d->_play->stop(); d->_play->start(); Arts::disconnect( d->_artssender, d->_bs2a ); d->_artssender = Arts::ByteSoundProducerV2::null(); d->_sender = 0; emit running( d->_attached ); } } void KAudioPlayStream::write( TQByteArray& ) { } void KAudioPlayStream::fillData( Arts::DataPacket<Arts::mcopbyte> *packet ) { //kdDebug( 400 ) << k_funcinfo << "packet->size=" << packet->size << endl; if ( d->_polling ) { TQByteArray bytearray( packet->size ); bytearray.setRawData( ( char* )packet->contents, packet->size ); bytearray.fill( 0 ); emit requestData( bytearray ); bytearray.resetRawData( ( char* )packet->contents, packet->size ); //for ( int i=0; i<10; i++ ) // kdDebug() << packet->contents[ i ] << " : " << bytearray.data()[ i ] << endl; } else { /// TODO: Implement a queue and fetching from it... } } // * * * KByteSoundProducer * * * KByteSoundProducer::KByteSoundProducer( KAudioPlayStream* impl, float minBufferTime, int rate, int bits, int channels, const char * title ) : _samplingRate( rate ) , _channels( channels ) , _bits( bits ) , _packets( 7 ) , _title( title ) , _impl( impl ) { // Calculate packet count (packetsize is fixed to packetCapacity = 4096 float streamBufferTime; do { _packets++; streamBufferTime = ( float )( _packets * packetCapacity * 1000 ) / ( float )( _samplingRate * _channels * 2 ); } while ( streamBufferTime < minBufferTime ); //kdDebug( 400 ) << k_funcinfo << "_packets:" << _packets << " packetCapacity:" << packetCapacity << endl; } KByteSoundProducer::~KByteSoundProducer() { } void KByteSoundProducer::streamStart() { outdata.setPull( _packets, packetCapacity ); } void KByteSoundProducer::streamEnd() { outdata.endPull(); } void KByteSoundProducer::request_outdata( Arts::DataPacket<Arts::mcopbyte> *packet ) { if ( _impl->running() ) { _impl->fillData( packet ); packet->send(); } } // vim: sw=4 ts=4 tw=80 #include "kaudioplaystream.moc" #include "kaudioplaystream_p.moc"