diff options
| author | trialuser02 <trialuser02@90c681e8-e032-0410-971d-27865f9a5e38> | 2008-03-23 17:25:56 +0000 |
|---|---|---|
| committer | trialuser02 <trialuser02@90c681e8-e032-0410-971d-27865f9a5e38> | 2008-03-23 17:25:56 +0000 |
| commit | 6a0e9d9d815674bedccbcc8e4ba6c25dc52fa3df (patch) | |
| tree | f99836a9179af7bb63d3bf78de9a9df6ad9ef86c /src/plugins/Output/pulseaudio/outputpulseaudio.cpp | |
| parent | 43587c89e00a7290c136c9bd6263c59fc2a2b120 (diff) | |
| download | qmmp-6a0e9d9d815674bedccbcc8e4ba6c25dc52fa3df.tar.gz qmmp-6a0e9d9d815674bedccbcc8e4ba6c25dc52fa3df.tar.bz2 qmmp-6a0e9d9d815674bedccbcc8e4ba6c25dc52fa3df.zip | |
Pulse Audio output plugin
git-svn-id: http://svn.code.sf.net/p/qmmp-dev/code/trunk/qmmp@292 90c681e8-e032-0410-971d-27865f9a5e38
Diffstat (limited to 'src/plugins/Output/pulseaudio/outputpulseaudio.cpp')
| -rw-r--r-- | src/plugins/Output/pulseaudio/outputpulseaudio.cpp | 235 |
1 files changed, 235 insertions, 0 deletions
diff --git a/src/plugins/Output/pulseaudio/outputpulseaudio.cpp b/src/plugins/Output/pulseaudio/outputpulseaudio.cpp new file mode 100644 index 000000000..02c4f52be --- /dev/null +++ b/src/plugins/Output/pulseaudio/outputpulseaudio.cpp @@ -0,0 +1,235 @@ +/*************************************************************************** + * Copyright (C) 2006-2008 by Ilya Kotov * + * forkotov02@hotmail.ru * + * * + * This program is free software; you can redistribute it and/or modify * + * it under the terms of the GNU General Public License as published by * + * the Free Software Foundation; either version 2 of the License, or * + * (at your option) any later version. * + * * + * This program 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 General Public License for more details. * + * * + * You should have received a copy of the GNU General Public License * + * along with this program; if not, write to the * + * Free Software Foundation, Inc., * + * 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA. * + ***************************************************************************/ + +#include <QObject> +#include <QApplication> +#include <QtGlobal> + +extern "C" { +#include <pulse/error.h> +} + +#include <stdio.h> +#include <string.h> +#include <iostream> + +#include <qmmp/constants.h> +#include <qmmp/buffer.h> +#include <qmmp/visual.h> +#include "outputpulseaudio.h" + +OutputPulseAudio::OutputPulseAudio(QObject * parent, bool useVolume) + : Output(parent), m_inited(FALSE), m_pause(FALSE), m_play(FALSE), + m_userStop(FALSE), m_totalWritten(0), m_currentSeconds(-1), + m_bps(-1), m_frequency(-1), m_channels(-1), m_precision(-1) +{ + Q_UNUSED(useVolume); + m_connection = 0; +} + +OutputPulseAudio::~OutputPulseAudio() +{ + uninitialize(); +} + +void OutputPulseAudio::stop() +{ + m_userStop = TRUE; +} + +void OutputPulseAudio::status() +{ + long ct = (m_totalWritten - latency()) / m_bps; + + if (ct < 0) + ct = 0; + + if (ct > m_currentSeconds) + { + m_currentSeconds = ct; + dispatch(m_currentSeconds, m_totalWritten, m_rate, + m_frequency, m_precision, m_channels); + } +} + +long OutputPulseAudio::written() +{ + return m_totalWritten; +} + +void OutputPulseAudio::seek(long pos) +{ + m_totalWritten = (pos * m_bps); + m_currentSeconds = -1; +} + +void OutputPulseAudio::configure(long freq, int chan, int prec, int brate) +{ + m_frequency = freq; + m_channels = chan; + m_precision = prec; + m_bps = freq * chan * (prec / 8); + + pa_sample_spec ss; + ss.format = PA_SAMPLE_S16LE; + ss.channels = chan; + ss.rate = freq; + int error; + m_connection = pa_simple_new(NULL, // Use the default server. + "Qmmp", // Our application's name. + PA_STREAM_PLAYBACK, + NULL, // Use the default device. + "Music", // Description of our stream. + &ss, // Our sample format. + NULL, // Use default channel map + NULL, // Use default buffering attributes. + &error // Error code. + ); + if (!m_connection) + { + qWarning("OutputPulseAudio: pa_simple_new() failed: %s", pa_strerror(error)); + m_inited = FALSE; + return; + } + qDebug("OutputPulseAudio: frequency=%d, channels=%d, bitrate=%d", uint(freq), chan, brate); +} + +void OutputPulseAudio::pause() +{ + if (!m_play) + return; + m_pause = (m_pause) ? FALSE : TRUE; + OutputState::Type state = m_pause ? OutputState::Paused: OutputState::Playing; + dispatch(state); +} + +bool OutputPulseAudio::initialize() +{ + m_inited = m_pause = m_play = m_userStop = FALSE; + m_currentSeconds = -1; + m_inited = TRUE; + return TRUE; +} + + +long OutputPulseAudio::latency() +{ + long used = 0; + + return used; +} + +void OutputPulseAudio::run() +{ + + mutex()->lock (); + if (! m_inited) + { + mutex()->unlock(); + return; + } + + m_play = TRUE; + + mutex()->unlock(); + + Buffer *b = 0; + bool done = FALSE; + int error; + + dispatch(OutputState::Playing); + + while (! done) + { + mutex()->lock (); + recycler()->mutex()->lock (); + + done = m_userStop; + + while (! done && (recycler()->empty() || m_pause)) + { + mutex()->unlock(); + recycler()->cond()->wakeOne(); + recycler()->cond()->wait(recycler()->mutex()); + mutex()->lock (); + done = m_userStop; + status(); + } + + if (! b) + { + b = recycler()->next(); + if (b->rate) + m_rate = b->rate; + } + + recycler()->cond()->wakeOne(); + recycler()->mutex()->unlock(); + + if (b) + { + if(pa_simple_write(m_connection, b->data, b->nbytes, &error) < 0) + { + mutex()->unlock(); + qWarning("OutputPulseAudio: pa_simple_write() failed: %s", pa_strerror(error)); + break; + } + + dispatchVisual(b, m_totalWritten, m_channels, m_precision); + status(); + m_totalWritten += b->nbytes; + mutex()->unlock(); + } + // force buffer change + recycler()->mutex()->lock (); + recycler()->done(); + recycler()->mutex()->unlock(); + b = 0; + } + + mutex()->lock (); + m_play = FALSE; + dispatch(OutputState::Stopped); + mutex()->unlock(); + +} + +void OutputPulseAudio::uninitialize() +{ + if (!m_inited) + return; + m_inited = FALSE; + m_pause = FALSE; + m_play = FALSE; + m_userStop = FALSE; + m_totalWritten = 0; + m_currentSeconds = -1; + m_bps = -1; + m_frequency = -1; + m_channels = -1; + m_precision = -1; + if (!m_connection) + { + qDebug("OutputPulseAudio: closing connection"); + pa_simple_free(m_connection); + m_connection = 0; + } + dispatch(OutputState::Stopped); +} |
