Mercurial > hg > svapp
view audioio/AudioPortAudioTarget.cpp @ 16:ba31fa322f93
* Fix a crash in PortAudio output for mono data
* Don't construct a chromagram in TonalChangeDetect plugin ctor
* Update layer show/hide/dormancy for multi-view-capable layers -- the
dormancy concept doesn't really work with these constraints any more
author | Chris Cannam |
---|---|
date | Fri, 03 Mar 2006 17:52:21 +0000 |
parents | df5923e33d01 |
children | c606d3ffa397 |
line wrap: on
line source
/* -*- c-basic-offset: 4 -*- vi:set ts=8 sts=4 sw=4: */ /* A waveform viewer and audio annotation editor. Chris Cannam, Queen Mary University of London, 2005-2006 This is experimental software. Not for distribution. */ #ifdef HAVE_PORTAUDIO #include "AudioPortAudioTarget.h" #include "AudioCallbackPlaySource.h" #include <iostream> #include <cassert> #include <cmath> //#define DEBUG_AUDIO_PORT_AUDIO_TARGET 1 AudioPortAudioTarget::AudioPortAudioTarget(AudioCallbackPlaySource *source) : AudioCallbackPlayTarget(source), m_stream(0), m_bufferSize(0), m_sampleRate(0), m_latency(0) { PaError err; err = Pa_Initialize(); if (err != paNoError) { std::cerr << "ERROR: AudioPortAudioTarget: Failed to initialize PortAudio" << std::endl; return; } m_bufferSize = 1024; m_sampleRate = 44100; if (m_source && (m_source->getSourceSampleRate() != 0)) { m_sampleRate = m_source->getSourceSampleRate(); } m_latency = Pa_GetMinNumBuffers(m_bufferSize, m_sampleRate) * m_bufferSize; err = Pa_OpenDefaultStream(&m_stream, 0, 2, paFloat32, m_sampleRate, m_bufferSize, 0, processStatic, this); if (err != paNoError) { std::cerr << "ERROR: AudioPortAudioTarget: Failed to open PortAudio stream" << std::endl; m_stream = 0; Pa_Terminate(); return; } err = Pa_StartStream(m_stream); if (err != paNoError) { std::cerr << "ERROR: AudioPortAudioTarget: Failed to start PortAudio stream" << std::endl; Pa_CloseStream(m_stream); m_stream = 0; Pa_Terminate(); return; } if (m_source) { std::cerr << "AudioPortAudioTarget: block size " << m_bufferSize << std::endl; m_source->setTargetBlockSize(m_bufferSize); m_source->setTargetSampleRate(m_sampleRate); m_source->setTargetPlayLatency(m_latency); } } AudioPortAudioTarget::~AudioPortAudioTarget() { if (m_stream) { PaError err; err = Pa_CloseStream(m_stream); if (err != paNoError) { std::cerr << "ERROR: AudioPortAudioTarget: Failed to close PortAudio stream" << std::endl; } Pa_Terminate(); } } bool AudioPortAudioTarget::isOK() const { return (m_stream != 0); } int AudioPortAudioTarget::processStatic(void *input, void *output, unsigned long nframes, PaTimestamp outTime, void *data) { return ((AudioPortAudioTarget *)data)->process(input, output, nframes, outTime); } void AudioPortAudioTarget::sourceModelReplaced() { m_source->setTargetSampleRate(m_sampleRate); } int AudioPortAudioTarget::process(void *inputBuffer, void *outputBuffer, unsigned long nframes, PaTimestamp) { #ifdef DEBUG_AUDIO_PORT_AUDIO_TARGET std::cout << "AudioPortAudioTarget::process(" << nframes << ")" << std::endl; #endif if (!m_source) return 0; float *output = (float *)outputBuffer; assert(nframes <= m_bufferSize); static float **tmpbuf = 0; static size_t tmpbufch = 0; static size_t tmpbufsz = 0; size_t sourceChannels = m_source->getSourceChannelCount(); // Because we offer pan, we always want at least 2 channels if (sourceChannels < 2) sourceChannels = 2; if (!tmpbuf || tmpbufch != sourceChannels || tmpbufsz < m_bufferSize) { if (tmpbuf) { for (size_t i = 0; i < tmpbufch; ++i) { delete[] tmpbuf[i]; } delete[] tmpbuf; } tmpbufch = sourceChannels; tmpbufsz = m_bufferSize; tmpbuf = new float *[tmpbufch]; for (size_t i = 0; i < tmpbufch; ++i) { tmpbuf[i] = new float[tmpbufsz]; } } m_source->getSourceSamples(nframes, tmpbuf); float peakLeft = 0.0, peakRight = 0.0; for (size_t ch = 0; ch < 2; ++ch) { float peak = 0.0; if (ch < sourceChannels) { // PortAudio samples are interleaved for (size_t i = 0; i < nframes; ++i) { output[i * 2 + ch] = tmpbuf[ch][i] * m_outputGain; float sample = fabsf(output[i * 2 + ch]); if (sample > peak) peak = sample; } } else if (ch == 1 && sourceChannels == 1) { for (size_t i = 0; i < nframes; ++i) { output[i * 2 + ch] = tmpbuf[0][i] * m_outputGain; float sample = fabsf(output[i * 2 + ch]); if (sample > peak) peak = sample; } } else { for (size_t i = 0; i < nframes; ++i) { output[i * 2 + ch] = 0; } } if (ch == 0) peakLeft = peak; if (ch > 0 || sourceChannels == 1) peakRight = peak; } m_source->setOutputLevels(peakLeft, peakRight); return 0; } #ifdef INCLUDE_MOCFILES #include "AudioPortAudioTarget.moc.cpp" #endif #endif /* HAVE_PORTAUDIO */