2015-01-25 22:10:49 +01:00
|
|
|
/** @file
|
|
|
|
* @author Edouard DUPIN
|
|
|
|
* @copyright 2011, Edouard DUPIN, all right reserved
|
2017-01-05 21:28:23 +01:00
|
|
|
* @license MPL v2.0 (see license file)
|
2015-01-25 22:10:49 +01:00
|
|
|
*/
|
|
|
|
|
2016-10-02 21:41:55 +02:00
|
|
|
#include <audio/drain/Resampler.hpp>
|
|
|
|
#include <audio/drain/debug.hpp>
|
2015-01-25 22:10:49 +01:00
|
|
|
|
2015-04-10 23:00:13 +02:00
|
|
|
audio::drain::Resampler::Resampler() :
|
2015-01-25 22:10:49 +01:00
|
|
|
#ifdef HAVE_SPEEX_DSP_RESAMPLE
|
|
|
|
m_speexResampler(nullptr),
|
|
|
|
#endif
|
|
|
|
m_positionRead(0),
|
|
|
|
m_positionWrite(0) {
|
|
|
|
|
|
|
|
}
|
2015-01-29 21:46:01 +01:00
|
|
|
|
2015-04-10 23:00:13 +02:00
|
|
|
void audio::drain::Resampler::init() {
|
|
|
|
audio::drain::Algo::init();
|
2015-02-01 22:22:42 +01:00
|
|
|
m_type = "Resampler";
|
2017-08-28 00:08:17 +02:00
|
|
|
m_supportedFormat.pushBack(audio::format_int16);
|
2015-04-13 21:49:48 +02:00
|
|
|
m_residualTimeInResampler = audio::Duration(0);
|
2015-01-29 21:46:01 +01:00
|
|
|
}
|
|
|
|
|
2016-07-19 21:43:58 +02:00
|
|
|
ememory::SharedPtr<audio::drain::Resampler> audio::drain::Resampler::create() {
|
|
|
|
ememory::SharedPtr<audio::drain::Resampler> tmp(new audio::drain::Resampler());
|
2015-01-29 21:46:01 +01:00
|
|
|
tmp->init();
|
|
|
|
return tmp;
|
|
|
|
}
|
|
|
|
|
2015-04-10 23:00:13 +02:00
|
|
|
audio::drain::Resampler::~Resampler() {
|
2015-01-25 22:10:49 +01:00
|
|
|
#ifdef HAVE_SPEEX_DSP_RESAMPLE
|
|
|
|
if (m_speexResampler != nullptr) {
|
|
|
|
speex_resampler_destroy(m_speexResampler);
|
|
|
|
m_speexResampler = nullptr;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
2015-04-10 23:00:13 +02:00
|
|
|
void audio::drain::Resampler::configurationChange() {
|
|
|
|
audio::drain::Algo::configurationChange();
|
2015-01-25 22:10:49 +01:00
|
|
|
if (m_input.getFormat() != m_output.getFormat()) {
|
2015-02-05 21:33:12 +01:00
|
|
|
DRAIN_ERROR("can not support Format Change ...");
|
2015-01-25 22:10:49 +01:00
|
|
|
m_needProcess = false;
|
|
|
|
}
|
2015-02-04 22:39:05 +01:00
|
|
|
if (m_input.getFormat() != audio::format_int16) {
|
2015-02-05 21:33:12 +01:00
|
|
|
DRAIN_ERROR("can not support Format other than int16_t ...");
|
2015-01-25 22:10:49 +01:00
|
|
|
m_needProcess = false;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
if (m_output.getMap() != m_output.getMap()) {
|
2015-02-05 21:33:12 +01:00
|
|
|
DRAIN_ERROR("can not support map Change ...");
|
2015-01-25 22:10:49 +01:00
|
|
|
m_needProcess = false;
|
|
|
|
}
|
|
|
|
if (m_input.getFrequency() == m_output.getFrequency()) {
|
|
|
|
// nothing to process...
|
|
|
|
m_needProcess = false;
|
|
|
|
return;
|
|
|
|
}
|
2015-02-13 21:06:55 +01:00
|
|
|
if ( m_input.getFrequency() == 0
|
|
|
|
|| m_output.getFrequency() == 0) {
|
|
|
|
DRAIN_WARNING("Configure IO with 0 frequency ... " << m_input << " to " << m_output);
|
|
|
|
return;
|
|
|
|
}
|
2015-01-25 22:10:49 +01:00
|
|
|
#ifdef HAVE_SPEEX_DSP_RESAMPLE
|
|
|
|
if (m_speexResampler != nullptr) {
|
|
|
|
speex_resampler_destroy(m_speexResampler);
|
|
|
|
m_speexResampler = nullptr;
|
|
|
|
}
|
|
|
|
int err = 0;
|
2015-02-13 21:06:55 +01:00
|
|
|
DRAIN_WARNING("Create resampler for : " << m_input << " to " << m_output);
|
2015-01-28 22:07:11 +01:00
|
|
|
m_speexResampler = speex_resampler_init(m_output.getMap().size(),
|
|
|
|
m_input.getFrequency(),
|
|
|
|
m_output.getFrequency(),
|
2015-01-25 22:10:49 +01:00
|
|
|
10, &err);
|
2015-04-13 21:49:48 +02:00
|
|
|
m_residualTimeInResampler = audio::Duration(0);
|
2015-01-25 22:10:49 +01:00
|
|
|
#else
|
2015-02-05 21:33:12 +01:00
|
|
|
DRAIN_WARNING("SPEEX DSP lib not accessible ==> can not resample");
|
2015-01-25 22:10:49 +01:00
|
|
|
m_needProcess = false;
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
2015-04-13 21:49:48 +02:00
|
|
|
bool audio::drain::Resampler::process(audio::Time& _time,
|
2015-04-10 23:00:13 +02:00
|
|
|
void* _input,
|
|
|
|
size_t _inputNbChunk,
|
|
|
|
void*& _output,
|
|
|
|
size_t& _outputNbChunk) {
|
2015-02-05 19:10:53 +01:00
|
|
|
drain::AutoLogInOut tmpLog("Resampler");
|
2015-01-25 22:10:49 +01:00
|
|
|
_outputNbChunk = 2048;
|
|
|
|
// chack if we need to process:
|
|
|
|
if (m_needProcess == false) {
|
2015-02-05 21:33:12 +01:00
|
|
|
DRAIN_WARNING("no process");
|
2015-01-25 22:10:49 +01:00
|
|
|
_output = _input;
|
|
|
|
_outputNbChunk = _inputNbChunk;
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
if (_input == nullptr) {
|
|
|
|
_output = &(m_outputData[0]);
|
|
|
|
_outputNbChunk = 0;
|
2015-02-05 21:33:12 +01:00
|
|
|
DRAIN_ERROR("null pointer input ... ");
|
2015-01-25 22:10:49 +01:00
|
|
|
return false;
|
|
|
|
}
|
2015-02-17 21:08:15 +01:00
|
|
|
// Update Output time with the previous delta of the buffer
|
|
|
|
DRAIN_VERBOSE("Resampler correct timestamp : " << _time << " ==> " << (_time - m_residualTimeInResampler));
|
|
|
|
_time -= m_residualTimeInResampler;
|
|
|
|
|
2015-04-13 21:49:48 +02:00
|
|
|
audio::Duration inTime(0, (int64_t(_inputNbChunk)*1000000000LL) / int64_t(m_input.getFrequency()));
|
2015-02-17 21:08:15 +01:00
|
|
|
m_residualTimeInResampler += inTime;
|
2015-01-25 22:10:49 +01:00
|
|
|
#ifdef HAVE_SPEEX_DSP_RESAMPLE
|
2015-01-28 22:07:11 +01:00
|
|
|
float nbInputTime = float(_inputNbChunk)/m_input.getFrequency();
|
2015-01-25 22:10:49 +01:00
|
|
|
float nbOutputSample = nbInputTime*m_output.getFrequency();
|
|
|
|
// we add 10% of the buffer size to have all the time enought data in the output to proceed all the input data...
|
2015-01-28 22:07:11 +01:00
|
|
|
_outputNbChunk = size_t(nbOutputSample*1.5f);
|
2015-02-05 21:33:12 +01:00
|
|
|
DRAIN_VERBOSE(" freq in=" << m_input.getFrequency() << " out=" << m_output.getFrequency());
|
|
|
|
DRAIN_VERBOSE(" Frame duration=" << nbInputTime);
|
|
|
|
DRAIN_VERBOSE(" nbInput chunk=" << _inputNbChunk << " nbOutputChunk=" << nbOutputSample);
|
2015-01-25 22:10:49 +01:00
|
|
|
|
2015-01-29 21:46:01 +01:00
|
|
|
m_outputData.resize(_outputNbChunk*m_output.getMap().size()*m_formatSize*16);
|
2015-01-25 22:10:49 +01:00
|
|
|
_output = &(m_outputData[0]);
|
|
|
|
if (m_speexResampler == nullptr) {
|
2015-02-05 21:33:12 +01:00
|
|
|
DRAIN_ERROR(" No speex resampler");
|
2015-01-25 22:10:49 +01:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
uint32_t nbChunkInput = _inputNbChunk;
|
|
|
|
uint32_t nbChunkOutput = _outputNbChunk;
|
2015-02-05 21:33:12 +01:00
|
|
|
DRAIN_VERBOSE(" >> input=" << nbChunkInput << " output=" << nbChunkOutput);
|
2015-01-25 22:10:49 +01:00
|
|
|
int ret = speex_resampler_process_interleaved_int(m_speexResampler,
|
|
|
|
static_cast<int16_t*>(_input),
|
|
|
|
&nbChunkInput,
|
|
|
|
static_cast<int16_t*>(_output),
|
|
|
|
&nbChunkOutput);
|
2015-02-05 21:33:12 +01:00
|
|
|
DRAIN_VERBOSE(" << input=" << nbChunkInput << " output=" << nbChunkOutput);
|
2015-01-25 22:10:49 +01:00
|
|
|
// update position of data:
|
|
|
|
m_positionWrite += nbChunkOutput;
|
|
|
|
// Check all input and output ...
|
|
|
|
if (nbChunkInput != _inputNbChunk) {
|
2015-02-05 21:33:12 +01:00
|
|
|
DRAIN_ERROR(" inputSize (not all read ...) proceed=" << nbChunkInput << " requested=" << _inputNbChunk);
|
2015-01-25 22:10:49 +01:00
|
|
|
// TODO : manage this case ...
|
|
|
|
}
|
|
|
|
if (nbChunkOutput == _outputNbChunk) {
|
2015-02-05 21:33:12 +01:00
|
|
|
DRAIN_ERROR(" Might have not enought data in output... output size=" << _outputNbChunk);
|
2015-01-25 22:10:49 +01:00
|
|
|
// TODO : manage this case ...
|
|
|
|
}
|
|
|
|
_outputNbChunk = nbChunkOutput;
|
2015-02-05 21:33:12 +01:00
|
|
|
DRAIN_VERBOSE(" process chunk=" << nbChunkInput << " out=" << nbChunkOutput);
|
2015-04-13 21:49:48 +02:00
|
|
|
audio::Duration outTime(0, (int64_t(_outputNbChunk)*1000000000LL) / int64_t(m_output.getFrequency()));
|
2017-09-26 15:57:44 +02:00
|
|
|
DRAIN_VERBOSE("convert " << _inputNbChunk << " ==> " << _outputNbChunk << " " << inTime << " => " << outTime);
|
2015-02-17 21:08:15 +01:00
|
|
|
// correct time :
|
|
|
|
m_residualTimeInResampler -= outTime;
|
|
|
|
/*
|
2017-09-26 15:57:44 +02:00
|
|
|
if (m_residualTimeInResampler.get() < 0) {
|
|
|
|
DRAIN_TODO("manage this case ... residual time in resampler : " << m_residualTimeInResampler << "ns");
|
2015-02-17 21:08:15 +01:00
|
|
|
}
|
|
|
|
*/
|
2015-01-25 22:10:49 +01:00
|
|
|
return true;
|
|
|
|
#else
|
|
|
|
_output = _input;
|
|
|
|
_outputNbChunk = _inputNbChunk;
|
|
|
|
return false;
|
|
|
|
#endif
|
|
|
|
}
|